1 /*
2 * Copyright (c) 2021 Huawei Device Co., Ltd.
3 * Licensed under the Apache License, Version 2.0 (the "License");
4 * you may not use this file except in compliance with the License.
5 * You may obtain a copy of the License at
6 *
7 * http://www.apache.org/licenses/LICENSE-2.0
8 *
9 * Unless required by applicable law or agreed to in writing, software
10 * distributed under the License is distributed on an "AS IS" BASIS,
11 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 * See the License for the specific language governing permissions and
13 * limitations under the License.
14 */
15
16 #include "le_task.h"
17
18
19 #include "le_loop.h"
20 #include "le_utils.h"
21
CheckTaskFlags(const BaseTask * task,uint32_t flags)22 int CheckTaskFlags(const BaseTask *task, uint32_t flags)
23 {
24 if (task == NULL) {
25 return 0;
26 }
27 return ((task->flags & flags) == flags);
28 }
29
GetSocketFd(const TaskHandle task)30 int GetSocketFd(const TaskHandle task)
31 {
32 BaseTask *stream = (BaseTask *)task;
33 return stream->taskId.fd;
34 }
35
CreateTask(const LoopHandle loopHandle,int fd,const LE_BaseInfo * info,uint32_t size)36 BaseTask *CreateTask(const LoopHandle loopHandle, int fd, const LE_BaseInfo *info, uint32_t size)
37 {
38 if ((size >= LOOP_MAX_BUFFER) || ((size + info->userDataSize) >= LOOP_MAX_BUFFER)) {
39 return NULL;
40 }
41 BaseTask *task = (BaseTask *)calloc(1, size + info->userDataSize);
42 LE_CHECK(task != NULL, return NULL, "Failed to alloc for task");
43 HASHMAPInitNode(&task->hashNode);
44 // key id
45 task->flags = info->flags;
46 task->taskId.fd = fd;
47 LE_STATUS ret = AddTask((EventLoop *)loopHandle, task);
48 LE_CHECK(ret == LE_SUCCESS, free(task);
49 return NULL, "Failed to alloc for task");
50 task->userDataSize = info->userDataSize;
51 task->userDataOffset = size;
52 task->close = info->close;
53 return task;
54 }
55
CloseTask(const LoopHandle loopHandle,BaseTask * task)56 void CloseTask(const LoopHandle loopHandle, BaseTask *task)
57 {
58 LE_CHECK(loopHandle != NULL && task != NULL, return, "Invalid parameters");
59 LE_LOGV("CloseTask %d", task->taskId.fd);
60 if (CheckTaskFlags(task, TASK_STREAM | TASK_CONNECT) ||
61 CheckTaskFlags(task, TASK_EVENT | TASK_ASYNC_EVENT)) {
62 StreamTask *stream = (StreamTask *)task;
63 LE_Buffer *buffer = GetFirstBuffer(stream);
64 while (buffer) {
65 FreeBuffer(loopHandle, stream, (BufferHandle)buffer);
66 buffer = GetFirstBuffer(stream);
67 }
68 }
69 if (task->close != NULL) {
70 task->close((TaskHandle)task);
71 }
72 }
73
CreateBuffer(uint32_t bufferSize)74 LE_Buffer *CreateBuffer(uint32_t bufferSize)
75 {
76 LE_ONLY_CHECK(bufferSize < LOOP_MAX_BUFFER, return NULL);
77 LE_Buffer *buffer = NULL;
78 LE_CHECK((buffer = (LE_Buffer *)malloc(sizeof(LE_Buffer) + bufferSize)) != NULL,
79 return NULL, "Failed to alloc memory for buffer");
80 OH_ListInit(&buffer->node);
81 buffer->buffSize = bufferSize;
82 buffer->dataSize = 0;
83 return buffer;
84 }
85
IsBufferEmpty(StreamTask * task)86 int IsBufferEmpty(StreamTask *task)
87 {
88 LoopMutexLock(&task->mutex);
89 int ret = ListEmpty(task->buffHead);
90 LoopMutexUnlock(&task->mutex);
91 return ret;
92 }
93
GetFirstBuffer(StreamTask * task)94 LE_Buffer *GetFirstBuffer(StreamTask *task)
95 {
96 LoopMutexLock(&task->mutex);
97 ListNode *node = task->buffHead.next;
98 LE_Buffer *buffer = NULL;
99 if (node != &task->buffHead) {
100 buffer = ListEntry(node, LE_Buffer, node);
101 }
102 LoopMutexUnlock(&task->mutex);
103 return buffer;
104 }
105
AddBuffer(StreamTask * task,LE_Buffer * buffer)106 void AddBuffer(StreamTask *task, LE_Buffer *buffer)
107 {
108 LoopMutexLock(&task->mutex);
109 OH_ListAddTail(&task->buffHead, &buffer->node);
110 LoopMutexUnlock(&task->mutex);
111 }
112
GetNextBuffer(StreamTask * task,const LE_Buffer * next)113 LE_Buffer *GetNextBuffer(StreamTask *task, const LE_Buffer *next)
114 {
115 LoopMutexLock(&task->mutex);
116 LE_Buffer *buffer = NULL;
117 ListNode *node = NULL;
118 if (next == NULL) {
119 node = task->buffHead.next;
120 } else {
121 node = next->node.next;
122 }
123 if (node != &task->buffHead) {
124 buffer = ListEntry(node, LE_Buffer, node);
125 }
126 LoopMutexUnlock(&task->mutex);
127 return buffer;
128 }
129
FreeBuffer(const LoopHandle loop,StreamTask * task,LE_Buffer * buffer)130 void FreeBuffer(const LoopHandle loop, StreamTask *task, LE_Buffer *buffer)
131 {
132 LE_CHECK(buffer != NULL, return, "Invalid buffer");
133 if (task == NULL) {
134 free(buffer);
135 return;
136 }
137 if (CheckTaskFlags((BaseTask *)task, TASK_STREAM | TASK_CONNECT) ||
138 CheckTaskFlags((BaseTask *)task, TASK_EVENT | TASK_ASYNC_EVENT)) {
139 LoopMutexLock(&task->mutex);
140 OH_ListRemove(&buffer->node);
141 LoopMutexUnlock(&task->mutex);
142 }
143 free(buffer);
144 }
145
LE_CreateBuffer(const LoopHandle loop,uint32_t bufferSize)146 BufferHandle LE_CreateBuffer(const LoopHandle loop, uint32_t bufferSize)
147 {
148 return (BufferHandle)CreateBuffer(bufferSize);
149 }
150
LE_FreeBuffer(const LoopHandle loop,const TaskHandle taskHandle,const BufferHandle handle)151 void LE_FreeBuffer(const LoopHandle loop, const TaskHandle taskHandle, const BufferHandle handle)
152 {
153 FreeBuffer(loop, (StreamTask *)taskHandle, (LE_Buffer *)handle);
154 }
155
LE_GetBufferInfo(const BufferHandle handle,uint32_t * dataSize,uint32_t * buffSize)156 uint8_t *LE_GetBufferInfo(const BufferHandle handle, uint32_t *dataSize, uint32_t *buffSize)
157 {
158 LE_Buffer *buffer = (LE_Buffer *)handle;
159 LE_CHECK(buffer != NULL, return NULL, "Invalid buffer");
160 if (dataSize) {
161 *dataSize = (uint32_t)buffer->dataSize;
162 }
163 if (buffSize) {
164 *buffSize = (uint32_t)buffer->buffSize;
165 }
166 return buffer->data;
167 }
168
LE_Send(const LoopHandle loopHandle,const TaskHandle taskHandle,const BufferHandle buffHandle,uint32_t buffLen)169 LE_STATUS LE_Send(const LoopHandle loopHandle,
170 const TaskHandle taskHandle, const BufferHandle buffHandle, uint32_t buffLen)
171 {
172 EventLoop *loop = (EventLoop *)loopHandle;
173 if (((BaseTask *)taskHandle)->flags & TASK_FLAGS_INVALID) {
174 LE_FreeBuffer(loopHandle, taskHandle, buffHandle);
175 return LE_INVALID_TASK;
176 }
177 LE_Buffer *buffer = (LE_Buffer *)buffHandle;
178 buffer->dataSize = buffLen;
179 if (CheckTaskFlags((BaseTask *)taskHandle, TASK_STREAM | TASK_CONNECT)) {
180 AddBuffer((StreamTask *)taskHandle, buffer);
181 } else if (CheckTaskFlags((BaseTask *)taskHandle, TASK_EVENT | TASK_ASYNC_EVENT)) {
182 AddBuffer((StreamTask *)taskHandle, buffer);
183 }
184 loop->modEvent(loop, (BaseTask *)taskHandle, EVENT_WRITE);
185 return LE_SUCCESS;
186 }
187
LE_CloseTask(const LoopHandle loopHandle,const TaskHandle taskHandle)188 void LE_CloseTask(const LoopHandle loopHandle, const TaskHandle taskHandle)
189 {
190 LE_CHECK(loopHandle != NULL && taskHandle != NULL, return, "Invalid parameters");
191 if (((LoopBase*)taskHandle)->flags & TASK_TIME) {
192 LE_StopTimer(loopHandle, taskHandle);
193 return;
194 }
195 LE_LOGV("LE_CloseTask %d", GetSocketFd(taskHandle));
196 BaseTask *task = (BaseTask *)taskHandle;
197 if (task->innerClose != NULL) {
198 task->innerClose(loopHandle, taskHandle);
199 }
200 free(task);
201 }
202
LE_GetUserData(TaskHandle handle)203 void *LE_GetUserData(TaskHandle handle)
204 {
205 LE_CHECK(handle != NULL, return NULL, "Invalid handle");
206 BaseTask *stream = (BaseTask *)handle;
207 return (void *)(((char *)stream) + stream->userDataOffset);
208 }
209
LE_GetSendResult(const BufferHandle handle)210 int32_t LE_GetSendResult(const BufferHandle handle)
211 {
212 LE_CHECK(handle != NULL, return 0, "Invalid handle");
213 return ((LE_Buffer *)handle)->result;
214 }