|
|
/* ----------------------------------------------------------------------------
|
|
|
* Copyright (c) Huawei Technologies Co., Ltd. 2013-2019. All rights reserved.
|
|
|
* Description: Queue
|
|
|
* Author: Huawei LiteOS Team
|
|
|
* Create: 2013-01-01
|
|
|
* Redistribution and use in source and binary forms, with or without modification,
|
|
|
* are permitted provided that the following conditions are met:
|
|
|
* 1. Redistributions of source code must retain the above copyright notice, this list of
|
|
|
* conditions and the following disclaimer.
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright notice, this list
|
|
|
* of conditions and the following disclaimer in the documentation and/or other materials
|
|
|
* provided with the distribution.
|
|
|
* 3. Neither the name of the copyright holder nor the names of its contributors may be used
|
|
|
* to endorse or promote products derived from this software without specific prior written
|
|
|
* permission.
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
|
|
|
* THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
|
|
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
|
|
|
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
|
|
|
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
|
|
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
|
|
|
* OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
|
|
|
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
|
|
|
* OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
|
|
|
* ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
* --------------------------------------------------------------------------- */
|
|
|
//实现了一个基于轻量级操作系统的队列管理模块,提供了队列的创建、读写、删除以及信息获取等功能
|
|
|
#include "los_queue_pri.h"
|
|
|
#include "los_queue_debug_pri.h"
|
|
|
#include "los_task_pri.h"
|
|
|
#include "los_spinlock.h"
|
|
|
#include "los_mp_pri.h"
|
|
|
#include "los_percpu_pri.h"
|
|
|
#include "los_trace.h"
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
#if __cplusplus
|
|
|
extern "C" {
|
|
|
#endif
|
|
|
#endif /* __cplusplus */
|
|
|
|
|
|
LITE_OS_SEC_BSS LosQueueCB *g_allQueue = NULL;
|
|
|
LITE_OS_SEC_BSS STATIC LOS_DL_LIST g_freeQueueList;
|
|
|
/* Error number for OsQueueBufferOperate */
|
|
|
#define OS_QUEUE_OPERATE_ERROR_INVALID_TYPE 1
|
|
|
#define OS_QUEUE_OPERATE_ERROR_MEMCPYS_GETMSG 2
|
|
|
#define OS_QUEUE_OPERATE_ERROR_MEMCPYS_MSG2BUF 3
|
|
|
#define OS_QUEUE_OPERATE_ERROR_MEMCPYS_STRMSG 4
|
|
|
#define OS_QUEUE_OPERATE_ERROR_MEMCPYS_STRMSGSIZE 5
|
|
|
|
|
|
/*
|
|
|
* Description : queue initial
|
|
|
* Return : LOS_OK on success or error code on failure
|
|
|
*/
|
|
|
//初始化队列,返回成功或失败错误码
|
|
|
LITE_OS_SEC_TEXT_INIT UINT32 OsQueueInit(VOID)
|
|
|
{
|
|
|
LosQueueCB *queueNode = NULL;
|
|
|
UINT32 index;
|
|
|
UINT32 size;
|
|
|
|
|
|
size = KERNEL_QUEUE_LIMIT * sizeof(LosQueueCB);
|
|
|
/* system resident memory, don't free */
|
|
|
g_allQueue = (LosQueueCB *)LOS_MemAlloc(m_aucSysMem0, size);
|
|
|
if (g_allQueue == NULL) {
|
|
|
return LOS_ERRNO_QUEUE_NO_MEMORY;
|
|
|
}
|
|
|
(VOID)memset_s(g_allQueue, size, 0, size);
|
|
|
LOS_ListInit(&g_freeQueueList);
|
|
|
for (index = 0; index < KERNEL_QUEUE_LIMIT; index++) {
|
|
|
queueNode = ((LosQueueCB *)g_allQueue) + index;
|
|
|
queueNode->queueId = index;
|
|
|
LOS_ListTailInsert(&g_freeQueueList, &queueNode->readWriteList[OS_QUEUE_WRITE]);
|
|
|
}
|
|
|
|
|
|
if (OsQueueDbgInitHook() != LOS_OK) {
|
|
|
return LOS_ERRNO_QUEUE_NO_MEMORY;
|
|
|
}
|
|
|
return LOS_OK;
|
|
|
}
|
|
|
//OsQueueCreateParameterCheck函数用于检查创建队列时传入的参数是否合法
|
|
|
STATIC INLINE UINT32 OsQueueCreateParameterCheck(UINT16 len, const UINT32 *queueId, UINT16 maxMsgSize)
|
|
|
{
|
|
|
if (queueId == NULL) {
|
|
|
return LOS_ERRNO_QUEUE_CREAT_PTR_NULL;
|
|
|
}
|
|
|
|
|
|
if (maxMsgSize > (OS_NULL_SHORT - sizeof(UINT32))) {
|
|
|
return LOS_ERRNO_QUEUE_SIZE_TOO_BIG;
|
|
|
}
|
|
|
|
|
|
if ((len == 0) || (maxMsgSize == 0)) {
|
|
|
return LOS_ERRNO_QUEUE_PARA_ISZERO;
|
|
|
}
|
|
|
return LOS_OK;
|
|
|
}
|
|
|
//OsQueueCreateInternal函数用于创建队列,包括在队列控制块中记录队列长度、消息大小等信息
|
|
|
LITE_OS_SEC_TEXT_INIT STATIC UINT32 OsQueueCreateInternal(UINT16 len, UINT32 *queueId, UINT16 msgSize,
|
|
|
UINT8 *queue, UINT8 queueMemType)
|
|
|
{
|
|
|
LosQueueCB *queueCB = NULL;
|
|
|
LOS_DL_LIST *unusedQueue = NULL;
|
|
|
UINT32 intSave;
|
|
|
|
|
|
SCHEDULER_LOCK(intSave);
|
|
|
if (LOS_ListEmpty(&g_freeQueueList)) {
|
|
|
SCHEDULER_UNLOCK(intSave);
|
|
|
OsQueueCheckHook();
|
|
|
return LOS_ERRNO_QUEUE_CB_UNAVAILABLE;
|
|
|
}
|
|
|
|
|
|
unusedQueue = LOS_DL_LIST_FIRST(&g_freeQueueList);
|
|
|
LOS_ListDelete(unusedQueue);
|
|
|
queueCB = GET_QUEUE_LIST(unusedQueue);
|
|
|
queueCB->queueLen = len;
|
|
|
queueCB->queueSize = msgSize;
|
|
|
queueCB->queueHandle = queue;
|
|
|
queueCB->queueState = LOS_USED;
|
|
|
queueCB->queueMemType = queueMemType;
|
|
|
queueCB->readWriteableCnt[OS_QUEUE_READ] = 0;
|
|
|
queueCB->readWriteableCnt[OS_QUEUE_WRITE] = len;
|
|
|
queueCB->queueHead = 0;
|
|
|
queueCB->queueTail = 0;
|
|
|
LOS_ListInit(&queueCB->readWriteList[OS_QUEUE_READ]);
|
|
|
LOS_ListInit(&queueCB->readWriteList[OS_QUEUE_WRITE]);
|
|
|
LOS_ListInit(&queueCB->memList);
|
|
|
|
|
|
OsQueueDbgUpdateHook(queueCB->queueId, OsCurrTaskGet()->taskEntry);
|
|
|
SCHEDULER_UNLOCK(intSave);
|
|
|
|
|
|
*queueId = queueCB->queueId;
|
|
|
|
|
|
LOS_TRACE(QUEUE_CREATE, *queueId, len, msgSize - sizeof(UINT32), (UINTPTR)queue, queueMemType);
|
|
|
return LOS_OK;
|
|
|
}
|
|
|
|
|
|
#ifdef LOSCFG_QUEUE_STATIC_ALLOCATION
|
|
|
LITE_OS_SEC_TEXT_INIT UINT32 LOS_QueueCreateStatic(const CHAR *queueName,
|
|
|
UINT16 len,
|
|
|
UINT32 *queueId,
|
|
|
UINT32 flags,
|
|
|
UINT16 maxMsgSize,
|
|
|
VOID *queueMem,
|
|
|
UINT16 memSize)
|
|
|
{
|
|
|
UINT32 ret;
|
|
|
UINT16 msgSize;
|
|
|
(VOID)queueName;
|
|
|
(VOID)flags;
|
|
|
|
|
|
ret = OsQueueCreateParameterCheck(len, queueId, maxMsgSize);
|
|
|
if (ret != LOS_OK) {
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
|
if (queueMem == NULL) {
|
|
|
return LOS_ERRNO_QUEUE_CREAT_PTR_NULL;
|
|
|
}
|
|
|
|
|
|
msgSize = maxMsgSize + sizeof(UINT32);
|
|
|
if (memSize < ((UINT32)msgSize * len)) {
|
|
|
return LOS_ERRNO_QUEUE_CREATE_NO_MEMORY;
|
|
|
}
|
|
|
|
|
|
return OsQueueCreateInternal(len, queueId, msgSize, queueMem, OS_QUEUE_ALLOC_STATIC);
|
|
|
}
|
|
|
#endif
|
|
|
//创建队列
|
|
|
LITE_OS_SEC_TEXT_INIT UINT32 LOS_QueueCreate(const CHAR *queueName, UINT16 len, UINT32 *queueId,
|
|
|
UINT32 flags, UINT16 maxMsgSize)
|
|
|
{
|
|
|
UINT32 ret;
|
|
|
UINT8 *queueMem = NULL;
|
|
|
UINT16 msgSize;
|
|
|
(VOID)queueName;
|
|
|
(VOID)flags;
|
|
|
|
|
|
ret = OsQueueCreateParameterCheck(len, queueId, maxMsgSize);
|
|
|
if (ret != LOS_OK) {
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
|
msgSize = maxMsgSize + sizeof(UINT32);
|
|
|
/*
|
|
|
* Memory allocation is time-consuming, to shorten the time of disable interrupt,
|
|
|
* move the memory allocation to here.
|
|
|
*/
|
|
|
queueMem = (UINT8 *)LOS_MemAlloc(m_aucSysMem1, (UINT32)len * msgSize);
|
|
|
if (queueMem == NULL) {
|
|
|
return LOS_ERRNO_QUEUE_CREATE_NO_MEMORY;
|
|
|
}
|
|
|
|
|
|
ret = OsQueueCreateInternal(len, queueId, msgSize, queueMem, OS_QUEUE_ALLOC_DYNAMIC);
|
|
|
if (ret != LOS_OK) {
|
|
|
(VOID)LOS_MemFree(m_aucSysMem1, queueMem);
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
|
return LOS_OK;
|
|
|
}
|
|
|
//OsQueueReadParameterCheck和OsQueueWriteParameterCheck分别用于检查读和写操作的参数合法性
|
|
|
LITE_OS_SEC_TEXT STATIC UINT32 OsQueueReadParameterCheck(UINT32 queueId, const VOID *bufferAddr,
|
|
|
const UINT32 *bufferSize, UINT32 timeout)
|
|
|
{
|
|
|
if (GET_QUEUE_INDEX(queueId) >= KERNEL_QUEUE_LIMIT) {
|
|
|
return LOS_ERRNO_QUEUE_INVALID;
|
|
|
}
|
|
|
|
|
|
if ((bufferAddr == NULL) || (bufferSize == NULL)) {
|
|
|
return LOS_ERRNO_QUEUE_READ_PTR_NULL;
|
|
|
}
|
|
|
|
|
|
if ((*bufferSize == 0) || (*bufferSize > (OS_NULL_SHORT - sizeof(UINT32)))) {
|
|
|
return LOS_ERRNO_QUEUE_READSIZE_IS_INVALID;
|
|
|
}
|
|
|
|
|
|
OsQueueDbgTimeUpdateHook(queueId);
|
|
|
|
|
|
if (timeout != LOS_NO_WAIT) {
|
|
|
if (OS_INT_ACTIVE) {
|
|
|
return LOS_ERRNO_QUEUE_READ_IN_INTERRUPT;
|
|
|
}
|
|
|
}
|
|
|
return LOS_OK;
|
|
|
}
|
|
|
|
|
|
LITE_OS_SEC_TEXT STATIC UINT32 OsQueueWriteParameterCheck(UINT32 queueId, const VOID *bufferAddr,
|
|
|
const UINT32 *bufferSize, UINT32 timeout)
|
|
|
{
|
|
|
if (GET_QUEUE_INDEX(queueId) >= KERNEL_QUEUE_LIMIT) {
|
|
|
return LOS_ERRNO_QUEUE_INVALID;
|
|
|
}
|
|
|
|
|
|
if (bufferAddr == NULL) {
|
|
|
return LOS_ERRNO_QUEUE_WRITE_PTR_NULL;
|
|
|
}
|
|
|
|
|
|
if (*bufferSize == 0) {
|
|
|
return LOS_ERRNO_QUEUE_WRITESIZE_ISZERO;
|
|
|
}
|
|
|
|
|
|
OsQueueDbgTimeUpdateHook(queueId);
|
|
|
|
|
|
if (timeout != LOS_NO_WAIT) {
|
|
|
if (OS_INT_ACTIVE) {
|
|
|
return LOS_ERRNO_QUEUE_WRITE_IN_INTERRUPT;
|
|
|
}
|
|
|
}
|
|
|
return LOS_OK;
|
|
|
}
|
|
|
//实现了队列的读取和写入操作,并对可能出现的错误进行了处理
|
|
|
STATIC UINT32 OsQueueBufferOperate(LosQueueCB *queueCB, UINT32 operateType, VOID *bufferAddr, UINT32 *bufferSize)
|
|
|
{
|
|
|
UINT8 *queueNode = NULL;
|
|
|
UINT32 msgDataSize;
|
|
|
UINT16 queuePosition;
|
|
|
|
|
|
/* get the queue position */
|
|
|
switch (OS_QUEUE_OPERATE_GET(operateType)) {
|
|
|
case OS_QUEUE_READ_HEAD:
|
|
|
queuePosition = queueCB->queueHead;
|
|
|
((queueCB->queueHead + 1) == queueCB->queueLen) ? (queueCB->queueHead = 0) : (queueCB->queueHead++);
|
|
|
break;
|
|
|
case OS_QUEUE_WRITE_HEAD:
|
|
|
(queueCB->queueHead == 0) ? (queueCB->queueHead = queueCB->queueLen - 1) : (--queueCB->queueHead);
|
|
|
queuePosition = queueCB->queueHead;
|
|
|
break;
|
|
|
case OS_QUEUE_WRITE_TAIL:
|
|
|
queuePosition = queueCB->queueTail;
|
|
|
((queueCB->queueTail + 1) == queueCB->queueLen) ? (queueCB->queueTail = 0) : (queueCB->queueTail++);
|
|
|
break;
|
|
|
default: /* read tail, reserved. */
|
|
|
return OS_QUEUE_OPERATE_ERROR_INVALID_TYPE;
|
|
|
}
|
|
|
|
|
|
queueNode = &(queueCB->queueHandle[(queuePosition * (queueCB->queueSize))]);
|
|
|
|
|
|
if (OS_QUEUE_IS_READ(operateType)) {
|
|
|
if (memcpy_s(&msgDataSize, sizeof(UINT32), queueNode + queueCB->queueSize - sizeof(UINT32),
|
|
|
sizeof(UINT32)) != EOK) {
|
|
|
return OS_QUEUE_OPERATE_ERROR_MEMCPYS_GETMSG;
|
|
|
}
|
|
|
if (memcpy_s(bufferAddr, *bufferSize, queueNode, msgDataSize) != EOK) {
|
|
|
return OS_QUEUE_OPERATE_ERROR_MEMCPYS_MSG2BUF;
|
|
|
}
|
|
|
|
|
|
*bufferSize = msgDataSize;
|
|
|
} else {
|
|
|
if (memcpy_s(queueNode, queueCB->queueSize, bufferAddr, *bufferSize) != EOK) {
|
|
|
return OS_QUEUE_OPERATE_ERROR_MEMCPYS_STRMSG;
|
|
|
}
|
|
|
if (memcpy_s(queueNode + queueCB->queueSize - sizeof(UINT32), sizeof(UINT32), bufferSize,
|
|
|
sizeof(UINT32)) != EOK) {
|
|
|
return OS_QUEUE_OPERATE_ERROR_MEMCPYS_STRMSGSIZE;
|
|
|
}
|
|
|
}
|
|
|
return LOS_OK;
|
|
|
}
|
|
|
//用于处理队列操作函数 OsQueueBufferOperate 返回的错误码,使得在发生错误时能够打印相应的错误信息
|
|
|
STATIC VOID OsQueueBufferOperateErrProcess(UINT32 errorCode)
|
|
|
{
|
|
|
switch (errorCode) {
|
|
|
case LOS_OK: break;
|
|
|
case OS_QUEUE_OPERATE_ERROR_INVALID_TYPE:
|
|
|
PRINT_ERR("invalid queue operate type!\n");
|
|
|
break;
|
|
|
case OS_QUEUE_OPERATE_ERROR_MEMCPYS_GETMSG:
|
|
|
PRINT_ERR("get msgdatasize failed\n");
|
|
|
break;
|
|
|
case OS_QUEUE_OPERATE_ERROR_MEMCPYS_MSG2BUF:
|
|
|
PRINT_ERR("copy message to buffer failed\n");
|
|
|
break;
|
|
|
case OS_QUEUE_OPERATE_ERROR_MEMCPYS_STRMSG:
|
|
|
PRINT_ERR("store message failed\n");
|
|
|
break;
|
|
|
case OS_QUEUE_OPERATE_ERROR_MEMCPYS_STRMSGSIZE:
|
|
|
PRINT_ERR("store message size failed\n");
|
|
|
break;
|
|
|
default:
|
|
|
PRINT_ERR("unknown queue operate ret %u\n", errorCode);
|
|
|
break;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
STATIC UINT32 OsQueueOperateParamCheck(const LosQueueCB *queueCB, UINT32 queueId,
|
|
|
UINT32 operateType, const UINT32 *bufferSize)
|
|
|
{
|
|
|
if ((queueCB->queueId != queueId) || (queueCB->queueState == LOS_UNUSED)) {
|
|
|
return LOS_ERRNO_QUEUE_NOT_CREATE;
|
|
|
}
|
|
|
|
|
|
if (OS_QUEUE_IS_READ(operateType) && (*bufferSize < (queueCB->queueSize - sizeof(UINT32)))) {
|
|
|
return LOS_ERRNO_QUEUE_READ_SIZE_TOO_SMALL;
|
|
|
} else if (OS_QUEUE_IS_WRITE(operateType) && (*bufferSize > (queueCB->queueSize - sizeof(UINT32)))) {
|
|
|
return LOS_ERRNO_QUEUE_WRITE_SIZE_TOO_BIG;
|
|
|
}
|
|
|
return LOS_OK;
|
|
|
}
|
|
|
//OsQueueOperate:实现了对队列的操作,包括参数检查、任务挂起与唤醒以及错误处理等功能
|
|
|
STATIC UINT32 OsQueueOperate(UINT32 queueId, UINT32 operateType, VOID *bufferAddr, UINT32 *bufferSize, UINT32 timeout)
|
|
|
{
|
|
|
LosQueueCB *queueCB = (LosQueueCB *)GET_QUEUE_HANDLE(queueId);
|
|
|
UINT32 readWrite = OS_QUEUE_READ_WRITE_GET(operateType);
|
|
|
UINT32 errorCode = LOS_OK;
|
|
|
UINT32 intSave, ret;
|
|
|
|
|
|
LOS_TRACE(QUEUE_RW, queueId, queueCB->queueSize, *bufferSize, operateType,
|
|
|
queueCB->readWriteableCnt[OS_QUEUE_READ], queueCB->readWriteableCnt[OS_QUEUE_WRITE], timeout);
|
|
|
|
|
|
SCHEDULER_LOCK(intSave);
|
|
|
ret = OsQueueOperateParamCheck(queueCB, queueId, operateType, bufferSize);
|
|
|
if (ret != LOS_OK) {
|
|
|
goto QUEUE_END;
|
|
|
}
|
|
|
|
|
|
if (queueCB->readWriteableCnt[readWrite] == 0) {
|
|
|
if (timeout == LOS_NO_WAIT) {
|
|
|
ret = OS_QUEUE_IS_READ(operateType) ? LOS_ERRNO_QUEUE_ISEMPTY : LOS_ERRNO_QUEUE_ISFULL;
|
|
|
goto QUEUE_END;
|
|
|
}
|
|
|
|
|
|
if (!OsPreemptableInSched()) {
|
|
|
ret = LOS_ERRNO_QUEUE_PEND_IN_LOCK;
|
|
|
goto QUEUE_END;
|
|
|
}
|
|
|
|
|
|
OsTaskWait(&queueCB->readWriteList[readWrite], OS_TASK_STATUS_PEND, timeout);
|
|
|
|
|
|
OsSchedResched();
|
|
|
SCHEDULER_UNLOCK(intSave);
|
|
|
SCHEDULER_LOCK(intSave);
|
|
|
|
|
|
if (OsCurrTaskGet()->taskStatus & OS_TASK_STATUS_TIMEOUT) {
|
|
|
OsCurrTaskGet()->taskStatus &= ~OS_TASK_STATUS_TIMEOUT;
|
|
|
ret = LOS_ERRNO_QUEUE_TIMEOUT;
|
|
|
goto QUEUE_END;
|
|
|
}
|
|
|
} else {
|
|
|
queueCB->readWriteableCnt[readWrite]--;
|
|
|
}
|
|
|
|
|
|
/* It will cause double lock issue that print after SCHEDULER_LOCK,
|
|
|
* so handle the return value errorCode after SCHEDULER_UNLOCK */
|
|
|
errorCode = OsQueueBufferOperate(queueCB, operateType, bufferAddr, bufferSize);
|
|
|
|
|
|
if (!LOS_ListEmpty(&queueCB->readWriteList[!readWrite])) {
|
|
|
LosTaskCB *resumedTask = OS_TCB_FROM_PENDLIST(LOS_DL_LIST_FIRST(&queueCB->readWriteList[!readWrite]));
|
|
|
OsTaskWake(resumedTask, OS_TASK_STATUS_PEND);
|
|
|
SCHEDULER_UNLOCK(intSave);
|
|
|
OsQueueBufferOperateErrProcess(errorCode);
|
|
|
LOS_MpSchedule(OS_MP_CPU_ALL);
|
|
|
LOS_Schedule();
|
|
|
return LOS_OK;
|
|
|
} else {
|
|
|
queueCB->readWriteableCnt[!readWrite]++;
|
|
|
}
|
|
|
|
|
|
QUEUE_END:
|
|
|
SCHEDULER_UNLOCK(intSave);
|
|
|
OsQueueBufferOperateErrProcess(errorCode);
|
|
|
return ret;
|
|
|
}
|
|
|
//LOS_QueueReadCopy:从队列中读取数据,通过参数检查后调用OsQueueOperate函数进行实际的读取操作
|
|
|
LITE_OS_SEC_TEXT UINT32 LOS_QueueReadCopy(UINT32 queueId,
|
|
|
VOID *bufferAddr,
|
|
|
UINT32 *bufferSize,
|
|
|
UINT32 timeout)
|
|
|
{
|
|
|
UINT32 ret;
|
|
|
UINT32 operateType;
|
|
|
|
|
|
ret = OsQueueReadParameterCheck(queueId, bufferAddr, bufferSize, timeout);
|
|
|
if (ret != LOS_OK) {
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
|
operateType = OS_QUEUE_OPERATE_TYPE(OS_QUEUE_READ, OS_QUEUE_HEAD);
|
|
|
return OsQueueOperate(queueId, operateType, bufferAddr, bufferSize, timeout);
|
|
|
}
|
|
|
//LOS_QueueWriteHeadCopy:向队列的头部写入数据,通过参数检查后调用OsQueueOperate函数进行实际的写入操作
|
|
|
LITE_OS_SEC_TEXT UINT32 LOS_QueueWriteHeadCopy(UINT32 queueId,
|
|
|
VOID *bufferAddr,
|
|
|
UINT32 bufferSize,
|
|
|
UINT32 timeout)
|
|
|
{
|
|
|
UINT32 ret;
|
|
|
UINT32 operateType;
|
|
|
|
|
|
ret = OsQueueWriteParameterCheck(queueId, bufferAddr, &bufferSize, timeout);
|
|
|
if (ret != LOS_OK) {
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
|
operateType = OS_QUEUE_OPERATE_TYPE(OS_QUEUE_WRITE, OS_QUEUE_HEAD);
|
|
|
return OsQueueOperate(queueId, operateType, bufferAddr, &bufferSize, timeout);
|
|
|
}
|
|
|
//LOS_QueueWriteCopy:向队列的尾部写入数据,通过参数检查后调用OsQueueOperate函数进行实际的写入操作
|
|
|
LITE_OS_SEC_TEXT UINT32 LOS_QueueWriteCopy(UINT32 queueId,
|
|
|
VOID *bufferAddr,
|
|
|
UINT32 bufferSize,
|
|
|
UINT32 timeout)
|
|
|
{
|
|
|
UINT32 ret;
|
|
|
UINT32 operateType;
|
|
|
|
|
|
ret = OsQueueWriteParameterCheck(queueId, bufferAddr, &bufferSize, timeout);
|
|
|
if (ret != LOS_OK) {
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
|
operateType = OS_QUEUE_OPERATE_TYPE(OS_QUEUE_WRITE, OS_QUEUE_TAIL);
|
|
|
return OsQueueOperate(queueId, operateType, bufferAddr, &bufferSize, timeout);
|
|
|
}
|
|
|
//LOS_QueueRead:从队列中读取数据的简化接口,直接调用LOS_QueueReadCopy函数
|
|
|
LITE_OS_SEC_TEXT UINT32 LOS_QueueRead(UINT32 queueId, VOID *bufferAddr, UINT32 bufferSize, UINT32 timeout)
|
|
|
{
|
|
|
return LOS_QueueReadCopy(queueId, bufferAddr, &bufferSize, timeout);
|
|
|
}
|
|
|
//LOS_QueueWrite:向队列中写入数据的简化接口,先进行参数检查,然后调用LOS_QueueWriteCopy函数
|
|
|
LITE_OS_SEC_TEXT UINT32 LOS_QueueWrite(UINT32 queueId, VOID *bufferAddr, UINT32 bufferSize, UINT32 timeout)
|
|
|
{
|
|
|
if (bufferAddr == NULL) {
|
|
|
return LOS_ERRNO_QUEUE_WRITE_PTR_NULL;
|
|
|
}
|
|
|
bufferSize = sizeof(CHAR *);
|
|
|
return LOS_QueueWriteCopy(queueId, &bufferAddr, bufferSize, timeout);
|
|
|
}
|
|
|
//LOS_QueueWriteHead:向队列头部写入数据的简化接口,先进行参数检查,然后调用LOS_QueueWriteHeadCopy函数
|
|
|
LITE_OS_SEC_TEXT UINT32 LOS_QueueWriteHead(UINT32 queueId,
|
|
|
VOID *bufferAddr,
|
|
|
UINT32 bufferSize,
|
|
|
UINT32 timeout)
|
|
|
{
|
|
|
if (bufferAddr == NULL) {
|
|
|
return LOS_ERRNO_QUEUE_WRITE_PTR_NULL;
|
|
|
}
|
|
|
bufferSize = sizeof(CHAR *);
|
|
|
return LOS_QueueWriteHeadCopy(queueId, &bufferAddr, bufferSize, timeout);
|
|
|
}
|
|
|
//LOS_QueueDelete:删除队列的函数
|
|
|
//首先检查队列是否存在,如果存在则检查是否有任务正在使用队列或者队列中还有未释放的内存块,如果满足条件则释放队列所占用的资源
|
|
|
LITE_OS_SEC_TEXT_INIT UINT32 LOS_QueueDelete(UINT32 queueId)
|
|
|
{
|
|
|
LosQueueCB *queueCB = NULL;
|
|
|
UINT8 *queue = NULL;
|
|
|
UINT32 intSave;
|
|
|
UINT32 ret = LOS_OK;
|
|
|
|
|
|
if (GET_QUEUE_INDEX(queueId) >= KERNEL_QUEUE_LIMIT) {
|
|
|
return LOS_ERRNO_QUEUE_NOT_FOUND;
|
|
|
}
|
|
|
|
|
|
queueCB = (LosQueueCB *)GET_QUEUE_HANDLE(queueId);
|
|
|
|
|
|
LOS_TRACE(QUEUE_DELETE, queueId, queueCB->queueState, queueCB->readWriteableCnt[OS_QUEUE_READ]);
|
|
|
|
|
|
SCHEDULER_LOCK(intSave);
|
|
|
if ((queueCB->queueId != queueId) || (queueCB->queueState == LOS_UNUSED)) {
|
|
|
ret = LOS_ERRNO_QUEUE_NOT_CREATE;
|
|
|
goto QUEUE_END;
|
|
|
}
|
|
|
|
|
|
if (!LOS_ListEmpty(&queueCB->readWriteList[OS_QUEUE_READ])) {
|
|
|
ret = LOS_ERRNO_QUEUE_IN_TSKUSE;
|
|
|
goto QUEUE_END;
|
|
|
}
|
|
|
|
|
|
if (!LOS_ListEmpty(&queueCB->readWriteList[OS_QUEUE_WRITE])) {
|
|
|
ret = LOS_ERRNO_QUEUE_IN_TSKUSE;
|
|
|
goto QUEUE_END;
|
|
|
}
|
|
|
|
|
|
if (!LOS_ListEmpty(&queueCB->memList)) {
|
|
|
ret = LOS_ERRNO_QUEUE_IN_TSKUSE;
|
|
|
goto QUEUE_END;
|
|
|
}
|
|
|
|
|
|
if ((queueCB->readWriteableCnt[OS_QUEUE_WRITE] + queueCB->readWriteableCnt[OS_QUEUE_READ]) !=
|
|
|
queueCB->queueLen) {
|
|
|
ret = LOS_ERRNO_QUEUE_IN_TSKWRITE;
|
|
|
goto QUEUE_END;
|
|
|
}
|
|
|
|
|
|
queue = queueCB->queueHandle;
|
|
|
queueCB->queueHandle = NULL;
|
|
|
queueCB->queueState = LOS_UNUSED;
|
|
|
queueCB->queueId = SET_QUEUE_ID(GET_QUEUE_COUNT(queueCB->queueId) + 1, GET_QUEUE_INDEX(queueCB->queueId));
|
|
|
OsQueueDbgUpdateHook(queueCB->queueId, NULL);
|
|
|
|
|
|
LOS_ListTailInsert(&g_freeQueueList, &queueCB->readWriteList[OS_QUEUE_WRITE]);
|
|
|
SCHEDULER_UNLOCK(intSave);
|
|
|
if (queueCB->queueMemType == OS_QUEUE_ALLOC_DYNAMIC) {
|
|
|
ret = LOS_MemFree(m_aucSysMem1, (VOID *)queue);
|
|
|
}
|
|
|
return ret;
|
|
|
|
|
|
QUEUE_END:
|
|
|
SCHEDULER_UNLOCK(intSave);
|
|
|
return ret;
|
|
|
}
|
|
|
//LOS_QueueInfoGet:获取指定队列的相关信息,包括队列ID、队列长度、队列大小、队列头尾指针、等待读取和写入的任务数等
|
|
|
LITE_OS_SEC_TEXT_MINOR UINT32 LOS_QueueInfoGet(UINT32 queueId, QUEUE_INFO_S *queueInfo)
|
|
|
{
|
|
|
UINT32 intSave;
|
|
|
UINT32 ret = LOS_OK;
|
|
|
LosQueueCB *queueCB = NULL;
|
|
|
LosTaskCB *tskCB = NULL;
|
|
|
|
|
|
if (queueInfo == NULL) {
|
|
|
return LOS_ERRNO_QUEUE_PTR_NULL;
|
|
|
}
|
|
|
|
|
|
if (GET_QUEUE_INDEX(queueId) >= KERNEL_QUEUE_LIMIT) {
|
|
|
return LOS_ERRNO_QUEUE_INVALID;
|
|
|
}
|
|
|
|
|
|
(VOID)memset_s((VOID *)queueInfo, sizeof(QUEUE_INFO_S), 0, sizeof(QUEUE_INFO_S));
|
|
|
|
|
|
SCHEDULER_LOCK(intSave);
|
|
|
queueCB = (LosQueueCB *)GET_QUEUE_HANDLE(queueId);
|
|
|
if ((queueCB->queueId != queueId) || (queueCB->queueState == LOS_UNUSED)) {
|
|
|
ret = LOS_ERRNO_QUEUE_NOT_CREATE;
|
|
|
goto QUEUE_END;
|
|
|
}
|
|
|
|
|
|
queueInfo->uwQueueID = queueId;
|
|
|
queueInfo->usQueueLen = queueCB->queueLen;
|
|
|
queueInfo->usQueueSize = queueCB->queueSize;
|
|
|
queueInfo->usQueueHead = queueCB->queueHead;
|
|
|
queueInfo->usQueueTail = queueCB->queueTail;
|
|
|
queueInfo->usReadableCnt = queueCB->readWriteableCnt[OS_QUEUE_READ];
|
|
|
queueInfo->usWritableCnt = queueCB->readWriteableCnt[OS_QUEUE_WRITE];
|
|
|
|
|
|
LOS_DL_LIST_FOR_EACH_ENTRY(tskCB, &queueCB->readWriteList[OS_QUEUE_READ], LosTaskCB, pendList) {
|
|
|
queueInfo->uwWaitReadTask |= (1ULL << tskCB->taskId);
|
|
|
}
|
|
|
|
|
|
LOS_DL_LIST_FOR_EACH_ENTRY(tskCB, &queueCB->readWriteList[OS_QUEUE_WRITE], LosTaskCB, pendList) {
|
|
|
queueInfo->uwWaitWriteTask |= (1ULL << tskCB->taskId);
|
|
|
}
|
|
|
|
|
|
LOS_DL_LIST_FOR_EACH_ENTRY(tskCB, &queueCB->memList, LosTaskCB, pendList) {
|
|
|
queueInfo->uwWaitMemTask |= (1ULL << tskCB->taskId);
|
|
|
}
|
|
|
|
|
|
QUEUE_END:
|
|
|
SCHEDULER_UNLOCK(intSave);
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
|
#ifdef LOSCFG_COMPAT_CMSIS
|
|
|
/*
|
|
|
* Description : Mail allocate memory
|
|
|
* Input : queueId --- QueueID
|
|
|
* : mailPool --- The memory poll that stores the mail
|
|
|
* : timeout --- Expiry time. The value range is [0,LOS_WAIT_FOREVER]
|
|
|
* Return : pointer if success otherwise NULL
|
|
|
*/
|
|
|
//OsQueueMailAlloc函数用来从邮件池中分配一个邮件内存块
|
|
|
//如果没有可用的内存块,则根据超时时间挂起当前任务,并将其加入到等待内存块的任务链表中
|
|
|
//当有内存块可用时,会将任务从任务链表中移除,并返回分配的内存块指针
|
|
|
LITE_OS_SEC_TEXT VOID *OsQueueMailAlloc(UINT32 queueId, VOID *mailPool, UINT32 timeout)
|
|
|
{
|
|
|
VOID *mem = NULL;
|
|
|
LosQueueCB *queueCB = NULL;
|
|
|
LosTaskCB *runTask = NULL;
|
|
|
UINT32 intSave;
|
|
|
|
|
|
if (GET_QUEUE_INDEX(queueId) >= KERNEL_QUEUE_LIMIT) {
|
|
|
return NULL;
|
|
|
}
|
|
|
|
|
|
if (mailPool == NULL) {
|
|
|
return NULL;
|
|
|
}
|
|
|
|
|
|
queueCB = GET_QUEUE_HANDLE(queueId);
|
|
|
|
|
|
OsQueueDbgTimeUpdateHook(queueId);
|
|
|
|
|
|
if (timeout != LOS_NO_WAIT) {
|
|
|
if (OS_INT_ACTIVE) {
|
|
|
return NULL;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
SCHEDULER_LOCK(intSave);
|
|
|
if ((queueCB->queueId != queueId) || (queueCB->queueState == LOS_UNUSED)) {
|
|
|
goto END;
|
|
|
}
|
|
|
|
|
|
mem = LOS_MemboxAlloc(mailPool);
|
|
|
if (mem == NULL) {
|
|
|
if (timeout == LOS_NO_WAIT) {
|
|
|
goto END;
|
|
|
}
|
|
|
|
|
|
runTask = OsCurrTaskGet();
|
|
|
OsTaskWait(&queueCB->memList, OS_TASK_STATUS_PEND, timeout);
|
|
|
|
|
|
OsSchedResched();
|
|
|
SCHEDULER_UNLOCK(intSave);
|
|
|
SCHEDULER_LOCK(intSave);
|
|
|
|
|
|
if (runTask->taskStatus & OS_TASK_STATUS_TIMEOUT) {
|
|
|
runTask->taskStatus &= ~OS_TASK_STATUS_TIMEOUT;
|
|
|
goto END;
|
|
|
} else {
|
|
|
/*
|
|
|
* When enters the current branch, means the current task already got a available membox,
|
|
|
* so the runTsk->msg can not be NULL.
|
|
|
*/
|
|
|
mem = runTask->msg;
|
|
|
runTask->msg = NULL;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
END:
|
|
|
SCHEDULER_UNLOCK(intSave);
|
|
|
return mem;
|
|
|
}
|
|
|
|
|
|
/*
|
|
|
* Description : Mail free memory
|
|
|
* Input : queueId --- QueueID
|
|
|
* : mailPool --- The mail memory poll address
|
|
|
* : mailMem --- The mail memory block address
|
|
|
* Return : LOS_OK on success or error code on failure
|
|
|
*/
|
|
|
//OsQueueMailFree函数用来释放指定的邮件内存块,并将其加入到邮件池的空闲链表中
|
|
|
//如果有任务在等待内存块,则会从任务链表中唤醒一个任务,并将其从任务链表中移除
|
|
|
LITE_OS_SEC_TEXT UINT32 OsQueueMailFree(UINT32 queueId, VOID *mailPool, VOID *mailMem)
|
|
|
{
|
|
|
VOID *mem = NULL;
|
|
|
LosQueueCB *queueCB = NULL;
|
|
|
LosTaskCB *resumedTask = NULL;
|
|
|
UINT32 intSave;
|
|
|
|
|
|
if (GET_QUEUE_INDEX(queueId) >= KERNEL_QUEUE_LIMIT) {
|
|
|
return LOS_ERRNO_QUEUE_MAIL_HANDLE_INVALID;
|
|
|
}
|
|
|
|
|
|
if (mailPool == NULL) {
|
|
|
return LOS_ERRNO_QUEUE_MAIL_PTR_INVALID;
|
|
|
}
|
|
|
|
|
|
queueCB = GET_QUEUE_HANDLE(queueId);
|
|
|
|
|
|
OsQueueDbgTimeUpdateHook(queueId);
|
|
|
|
|
|
SCHEDULER_LOCK(intSave);
|
|
|
|
|
|
if (LOS_MemboxFree(mailPool, mailMem)) {
|
|
|
SCHEDULER_UNLOCK(intSave);
|
|
|
return LOS_ERRNO_QUEUE_MAIL_FREE_ERROR;
|
|
|
}
|
|
|
|
|
|
if ((queueCB->queueId != queueId) || (queueCB->queueState == LOS_UNUSED)) {
|
|
|
SCHEDULER_UNLOCK(intSave);
|
|
|
return LOS_ERRNO_QUEUE_NOT_CREATE;
|
|
|
}
|
|
|
|
|
|
OsQueueDbgTimeUpdateHook(queueId);
|
|
|
|
|
|
if (!LOS_ListEmpty(&queueCB->memList)) {
|
|
|
resumedTask = OS_TCB_FROM_PENDLIST(LOS_DL_LIST_FIRST(&queueCB->memList));
|
|
|
OsTaskWake(resumedTask, OS_TASK_STATUS_PEND);
|
|
|
mem = LOS_MemboxAlloc(mailPool);
|
|
|
/* At the state of LOS_IntLock, the allocation can not be failed after releasing succefully. */
|
|
|
resumedTask->msg = mem;
|
|
|
SCHEDULER_UNLOCK(intSave);
|
|
|
LOS_MpSchedule(OS_MP_CPU_ALL);
|
|
|
LOS_Schedule();
|
|
|
} else {
|
|
|
SCHEDULER_UNLOCK(intSave);
|
|
|
}
|
|
|
return LOS_OK;
|
|
|
}
|
|
|
#endif /* LOSCFG_COMPAT_CMSIS */
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
#if __cplusplus
|
|
|
}
|
|
|
#endif
|
|
|
#endif /* __cplusplus */
|