/* * Copyright (c) 2013-2019 Huawei Technologies Co., Ltd. All rights reserved. * Copyright (c) 2020-2023 Huawei Device Co., Ltd. All rights reserved. * * 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 "mqueue.h" #ifdef LOSCFG_FS_VFS #include "fcntl.h" #include "pthread.h" #include "map_error.h" #include "time_posix.h" #include "los_memory.h" #include "los_vm_map.h" #include "los_process_pri.h" #include "fs/file.h" #include "user_copy.h" #define FNONBLOCK O_NONBLOCK #ifndef LOSCFG_IPC_CONTAINER /* GLOBALS */ STATIC fd_set g_queueFdSet; STATIC struct mqarray g_queueTable[LOSCFG_BASE_IPC_QUEUE_LIMIT]; STATIC pthread_mutex_t g_mqueueMutex = PTHREAD_RECURSIVE_MUTEX_INITIALIZER_NP; STATIC struct mqpersonal *g_mqPrivBuf[MAX_MQ_FD]; #define IPC_QUEUE_FD_SET g_queueFdSet #define IPC_QUEUE_TABLE g_queueTable #define IPC_QUEUE_MUTEX g_mqueueMutex #define IPC_QUEUE_MQ_PRIV_BUF g_mqPrivBuf #endif /* LOCAL FUNCTIONS */ STATIC INLINE INT32 MqNameCheck(const CHAR *mqName) { if (mqName == NULL) { errno = EINVAL; return -1; } if (strlen(mqName) == 0) { errno = EINVAL; return -1; } if (strlen(mqName) > (PATH_MAX - 1)) { errno = ENAMETOOLONG; return -1; } return 0; } STATIC INLINE UINT32 GetMqueueCBByID(UINT32 queueID, LosQueueCB **queueCB) { LosQueueCB *tmpQueueCB = NULL; if (queueCB == NULL) { errno = EINVAL; return LOS_ERRNO_QUEUE_READ_PTR_NULL; } tmpQueueCB = GET_QUEUE_HANDLE(queueID); if ((GET_QUEUE_INDEX(queueID) >= LOSCFG_BASE_IPC_QUEUE_LIMIT) || (tmpQueueCB->queueID != queueID)) { return LOS_ERRNO_QUEUE_INVALID; } *queueCB = tmpQueueCB; return LOS_OK; } STATIC INLINE struct mqarray *GetMqueueCBByName(const CHAR *name) { UINT32 index; UINT32 mylen = strlen(name); for (index = 0; index < LOSCFG_BASE_IPC_QUEUE_LIMIT; index++) { if ((IPC_QUEUE_TABLE[index].mq_name == NULL) || (strlen(IPC_QUEUE_TABLE[index].mq_name) != mylen)) { continue; } if (strncmp(name, (const CHAR *)(IPC_QUEUE_TABLE[index].mq_name), mylen) == 0) { return &(IPC_QUEUE_TABLE[index]); } } return NULL; } STATIC INT32 DoMqueueDelete(struct mqarray *mqueueCB) { UINT32 ret; if (mqueueCB->mq_name != NULL) { LOS_MemFree(OS_SYS_MEM_ADDR, mqueueCB->mq_name); mqueueCB->mq_name = NULL; } mqueueCB->mqcb = NULL; /* When mqueue-list head node needed free ,reset the mode_data */ mqueueCB->mode_data.data = 0; mqueueCB->euid = -1; mqueueCB->egid = -1; mqueueCB->mq_notify.pid = 0; ret = LOS_QueueDelete(mqueueCB->mq_id); switch (ret) { case LOS_OK: return 0; case LOS_ERRNO_QUEUE_NOT_FOUND: case LOS_ERRNO_QUEUE_NOT_CREATE: case LOS_ERRNO_QUEUE_IN_TSKUSE: case LOS_ERRNO_QUEUE_IN_TSKWRITE: errno = EAGAIN; return -1; default: errno = EINVAL; return -1; } } STATIC int SaveMqueueName(const CHAR *mqName, struct mqarray *mqueueCB) { size_t nameLen; nameLen = strlen(mqName); /* sys_mq_open has checked name and name length */ mqueueCB->mq_name = (char *)LOS_MemAlloc(OS_SYS_MEM_ADDR, nameLen + 1); if (mqueueCB->mq_name == NULL) { errno = ENOMEM; return LOS_NOK; } if (strncpy_s(mqueueCB->mq_name, (nameLen + 1), mqName, nameLen) != EOK) { LOS_MemFree(OS_SYS_MEM_ADDR, mqueueCB->mq_name); mqueueCB->mq_name = NULL; errno = EINVAL; return LOS_NOK; } mqueueCB->mq_name[nameLen] = '\0'; return LOS_OK; } STATIC struct mqpersonal *DoMqueueCreate(const struct mq_attr *attr, const CHAR *mqName, INT32 openFlag, UINT32 mode) { struct mqarray *mqueueCB = NULL; UINT32 mqueueID; UINT32 err = LOS_QueueCreate(NULL, attr->mq_maxmsg, &mqueueID, 0, attr->mq_msgsize); if (map_errno(err) != ENOERR) { goto ERROUT; } if (IPC_QUEUE_TABLE[GET_QUEUE_INDEX(mqueueID)].mqcb == NULL) { mqueueCB = &(IPC_QUEUE_TABLE[GET_QUEUE_INDEX(mqueueID)]); mqueueCB->mq_id = mqueueID; } if (mqueueCB == NULL) { errno = EINVAL; goto ERROUT; } if (SaveMqueueName(mqName, mqueueCB) != LOS_OK) { goto ERROUT; } if (GetMqueueCBByID(mqueueCB->mq_id, &(mqueueCB->mqcb)) != LOS_OK) { errno = ENOSPC; goto ERROUT; } mqueueCB->mq_personal = (struct mqpersonal *)LOS_MemAlloc(OS_SYS_MEM_ADDR, sizeof(struct mqpersonal)); if (mqueueCB->mq_personal == NULL) { (VOID)LOS_QueueDelete(mqueueCB->mq_id); mqueueCB->mqcb->queueHandle = NULL; mqueueCB->mqcb = NULL; errno = ENOSPC; goto ERROUT; } mqueueCB->unlinkflag = FALSE; mqueueCB->unlink_ref = 0; mqueueCB->mq_personal->mq_status = MQ_USE_MAGIC; mqueueCB->mq_personal->mq_next = NULL; mqueueCB->mq_personal->mq_posixdes = mqueueCB; mqueueCB->mq_personal->mq_flags = (INT32)((UINT32)openFlag | ((UINT32)attr->mq_flags & (UINT32)FNONBLOCK)); mqueueCB->mq_personal->mq_mode = mode; mqueueCB->mq_personal->mq_refcount = 0; mqueueCB->mq_notify.pid = 0; return mqueueCB->mq_personal; ERROUT: if ((mqueueCB != NULL) && (mqueueCB->mq_name != NULL)) { LOS_MemFree(OS_SYS_MEM_ADDR, mqueueCB->mq_name); mqueueCB->mq_name = NULL; } return (struct mqpersonal *)-1; } STATIC struct mqpersonal *DoMqueueOpen(struct mqarray *mqueueCB, INT32 openFlag) { struct mqpersonal *privateMqPersonal = NULL; /* already have the same name of g_squeuetable */ if (mqueueCB->unlinkflag == TRUE) { errno = EINVAL; goto ERROUT; } /* alloc mqprivate and add to mqarray */ privateMqPersonal = (struct mqpersonal *)LOS_MemAlloc(OS_SYS_MEM_ADDR, sizeof(struct mqpersonal)); if (privateMqPersonal == NULL) { errno = ENOSPC; goto ERROUT; } privateMqPersonal->mq_next = mqueueCB->mq_personal; mqueueCB->mq_personal = privateMqPersonal; privateMqPersonal->mq_posixdes = mqueueCB; privateMqPersonal->mq_flags = openFlag; privateMqPersonal->mq_status = MQ_USE_MAGIC; privateMqPersonal->mq_refcount = 0; return privateMqPersonal; ERROUT: return (struct mqpersonal *)-1; } STATIC INT32 DoMqueueClose(struct mqpersonal *privateMqPersonal) { struct mqarray *mqueueCB = NULL; struct mqpersonal *tmp = NULL; mqueueCB = privateMqPersonal->mq_posixdes; if (mqueueCB == NULL || mqueueCB->mq_personal == NULL) { errno = EBADF; return LOS_NOK; } /* find the personal and remove */ if (mqueueCB->mq_personal == privateMqPersonal) { mqueueCB->mq_personal = privateMqPersonal->mq_next; } else { for (tmp = mqueueCB->mq_personal; tmp->mq_next != NULL; tmp = tmp->mq_next) { if (tmp->mq_next == privateMqPersonal) { break; } } if (tmp->mq_next == NULL) { errno = EBADF; return LOS_NOK; } tmp->mq_next = privateMqPersonal->mq_next; } /* flag no use */ privateMqPersonal->mq_status = 0; /* free the personal */ (VOID)LOS_MemFree(OS_SYS_MEM_ADDR, privateMqPersonal); if ((mqueueCB->unlinkflag == TRUE) && (mqueueCB->mq_personal == NULL)) { return DoMqueueDelete(mqueueCB); } return LOS_OK; } /* Translate a sysFd into privateMqPersonal */ STATIC struct mqpersonal *MqGetPrivDataBuff(mqd_t personal) { INT32 sysFd = (INT32)personal; INT32 id = sysFd - MQUEUE_FD_OFFSET; /* Filter illegal id */ if ((id < 0) || (id >= MAX_MQ_FD)) { errno = EBADF; return NULL; } return IPC_QUEUE_MQ_PRIV_BUF[id]; } /** * Alloc sysFd, storage mq private data, set using bit. * * @param maxfdp: Maximum allowed application of mqueue sysFd. * @param fdset: Mqueue sysFd bit map. * @param privateMqPersonal: Private data. * @return the index of the new fd; -1 on error */ STATIC INT32 MqAllocSysFd(int maxfdp, struct mqpersonal *privateMqPersonal) { INT32 i; fd_set *fdset = &IPC_QUEUE_FD_SET; for (i = 0; i < maxfdp; i++) { /* sysFd: used bit setting, and get the index of swtmrID buffer */ if (fdset && !(FD_ISSET(i + MQUEUE_FD_OFFSET, fdset))) { FD_SET(i + MQUEUE_FD_OFFSET, fdset); if (!IPC_QUEUE_MQ_PRIV_BUF[i]) { IPC_QUEUE_MQ_PRIV_BUF[i] = privateMqPersonal; return i + MQUEUE_FD_OFFSET; } } } return -1; } STATIC VOID MqFreeSysFd(mqd_t personal) { INT32 sysFd = (INT32)personal; fd_set *fdset = &IPC_QUEUE_FD_SET; if (fdset && FD_ISSET(sysFd, fdset)) { FD_CLR(sysFd, fdset); IPC_QUEUE_MQ_PRIV_BUF[sysFd - MQUEUE_FD_OFFSET] = NULL; } } /* Mqueue fd reference count */ void MqueueRefer(int sysFd) { struct mqarray *mqueueCB = NULL; struct mqpersonal *privateMqPersonal = NULL; (VOID)pthread_mutex_lock(&IPC_QUEUE_MUTEX); /* Get the personal sysFd and reset personal fd -1 */ privateMqPersonal = MqGetPrivDataBuff((mqd_t)sysFd); if (privateMqPersonal == NULL) { goto OUT_UNLOCK; } mqueueCB = privateMqPersonal->mq_posixdes; if (mqueueCB == NULL) { goto OUT_UNLOCK; } privateMqPersonal->mq_refcount++; OUT_UNLOCK: (VOID)pthread_mutex_unlock(&IPC_QUEUE_MUTEX); return; } STATIC INT32 MqTryClose(struct mqpersonal *privateMqPersonal) { struct mqarray *mqueueCB = NULL; mqueueCB = privateMqPersonal->mq_posixdes; if (mqueueCB == NULL) { errno = ENFILE; return false; } if (privateMqPersonal->mq_refcount == 0) { return TRUE; } privateMqPersonal->mq_refcount--; return FALSE; } /* Set the mode data bit,for consumer's mode comparing. */ STATIC INT32 MqueueModeAnalysisSet(struct mqpersonal *privateMqPersonal) { UINT32 mode; UINT32 intSave; User *user = NULL; struct mqarray *mqueueCB = NULL; if ((INT32)(UINTPTR)privateMqPersonal < 0) { return -1; } /* Get mqueueCB of first time creating mqueue */ mqueueCB = privateMqPersonal->mq_posixdes; if (mqueueCB == NULL) { errno = ENFILE; return -1; } mode = mqueueCB->mq_personal->mq_mode; /* Set mqueue gid uid */ SCHEDULER_LOCK(intSave); user = OsCurrUserGet(); mqueueCB->euid = user->effUserID; mqueueCB->egid = user->effGid; SCHEDULER_UNLOCK(intSave); /* Set mode data bit */ if (mode & S_IRUSR) { mqueueCB->mode_data.usr |= S_IRUSR; } if (mode & S_IWUSR) { mqueueCB->mode_data.usr |= S_IWUSR; } if (mode & S_IRGRP) { mqueueCB->mode_data.grp |= S_IRGRP; } if (mode & S_IWGRP) { mqueueCB->mode_data.grp |= S_IWGRP; } if (mode & S_IROTH) { mqueueCB->mode_data.oth |= S_IROTH; } if (mode & S_IWOTH) { mqueueCB->mode_data.oth |= S_IWOTH; } return 0; } STATIC INT32 GetPermissionOfVisitor(struct mqarray *mqueueCB) { uid_t euid; gid_t egid; UINT32 intSave; User *user = NULL; if (mqueueCB == NULL) { errno = ENOENT; return -EPERM; } /* Get the visitor process euid and egid */ SCHEDULER_LOCK(intSave); user = OsCurrUserGet(); euid = user->effUserID; egid = user->effGid; SCHEDULER_UNLOCK(intSave); /* root */ if (euid == 0) { return ENOERR; } if (euid == mqueueCB->euid) { /* usr */ if (!((mqueueCB->mode_data.usr & S_IRUSR) || (mqueueCB->mode_data.usr & S_IWUSR))) { errno = EACCES; goto ERR_OUT; } } else if (egid == mqueueCB->egid) { /* grp */ if (!((mqueueCB->mode_data.grp & S_IRGRP) || (mqueueCB->mode_data.grp & S_IWGRP))) { errno = EACCES; goto ERR_OUT; } } else { /* oth */ if (!((mqueueCB->mode_data.oth & S_IROTH) || (mqueueCB->mode_data.oth & S_IWOTH))) { errno = EACCES; goto ERR_OUT; } } return ENOERR; ERR_OUT: return -EPERM; } STATIC INT32 GetMqueueAttr(struct mq_attr *defaultAttr, struct mq_attr *attr) { if (attr != NULL) { if (LOS_ArchCopyFromUser(defaultAttr, attr, sizeof(struct mq_attr))) { errno = EFAULT; return -1; } if ((defaultAttr->mq_maxmsg < 0) || (defaultAttr->mq_maxmsg > (long int)USHRT_MAX) || (defaultAttr->mq_msgsize < 0) || (defaultAttr->mq_msgsize > (long int)(USHRT_MAX - sizeof(UINT32)))) { errno = EINVAL; return -1; } } return 0; } mqd_t mq_open(const char *mqName, int openFlag, ...) { struct mqarray *mqueueCB = NULL; struct mqpersonal *privateMqPersonal = (struct mqpersonal *)-1; struct mq_attr *attr = NULL; struct mq_attr defaultAttr = { 0, MQ_MAX_MSG_NUM, MQ_MAX_MSG_LEN, 0 }; va_list ap; int sysFd; mqd_t mqFd = -1; unsigned int mode = 0; if (MqNameCheck(mqName) == -1) { return (mqd_t)-1; } (VOID)pthread_mutex_lock(&IPC_QUEUE_MUTEX); mqueueCB = GetMqueueCBByName(mqName); if ((UINT32)openFlag & (UINT32)O_CREAT) { if (mqueueCB != NULL) { if (((UINT32)openFlag & (UINT32)O_EXCL)) { errno = EEXIST; goto OUT; } privateMqPersonal = DoMqueueOpen(mqueueCB, openFlag); } else { va_start(ap, openFlag); mode = va_arg(ap, unsigned int); attr = va_arg(ap, struct mq_attr *); va_end(ap); if (GetMqueueAttr(&defaultAttr, attr)) { goto OUT; } privateMqPersonal = DoMqueueCreate(&defaultAttr, mqName, openFlag, mode); } /* Set mode data bit ,just for the first node */ if (MqueueModeAnalysisSet(privateMqPersonal)) { if ((INT32)(UINTPTR)privateMqPersonal > 0) { (VOID)DoMqueueClose(privateMqPersonal); } goto OUT; } } else { if (GetPermissionOfVisitor(mqueueCB)) { goto OUT; } privateMqPersonal = DoMqueueOpen(mqueueCB, openFlag); } if ((INT32)(UINTPTR)privateMqPersonal > 0) { /* alloc sysFd */ sysFd = MqAllocSysFd(MAX_MQ_FD, privateMqPersonal); if (sysFd == -1) { /* there are no more mq sysFd to use, close the personal */ (VOID)DoMqueueClose(privateMqPersonal); errno = ENFILE; } mqFd = (mqd_t)sysFd; } OUT: (VOID)pthread_mutex_unlock(&IPC_QUEUE_MUTEX); return mqFd; } int mq_close(mqd_t personal) { INT32 ret = -1; struct mqpersonal *privateMqPersonal = NULL; (VOID)pthread_mutex_lock(&IPC_QUEUE_MUTEX); /* Get the personal sysFd and reset personal fd -1 */ privateMqPersonal = MqGetPrivDataBuff(personal); if (privateMqPersonal == NULL) { goto OUT_UNLOCK; } if (privateMqPersonal->mq_status != MQ_USE_MAGIC) { errno = EBADF; goto OUT_UNLOCK; } if (!MqTryClose(privateMqPersonal)) { ret = 0; goto OUT_UNLOCK; } ret = DoMqueueClose(privateMqPersonal); if (ret < 0) { goto OUT_UNLOCK; } MqFreeSysFd(personal); OUT_UNLOCK: (VOID)pthread_mutex_unlock(&IPC_QUEUE_MUTEX); return ret; } int OsMqGetAttr(mqd_t personal, struct mq_attr *mqAttr) { struct mqarray *mqueueCB = NULL; struct mqpersonal *privateMqPersonal = NULL; privateMqPersonal = MqGetPrivDataBuff(personal); if (privateMqPersonal == NULL) { return -1; } if (mqAttr == NULL) { errno = EINVAL; return -1; } (VOID)pthread_mutex_lock(&IPC_QUEUE_MUTEX); if (privateMqPersonal->mq_status != MQ_USE_MAGIC) { errno = EBADF; (VOID)pthread_mutex_unlock(&IPC_QUEUE_MUTEX); return -1; } mqueueCB = privateMqPersonal->mq_posixdes; mqAttr->mq_maxmsg = mqueueCB->mqcb->queueLen; mqAttr->mq_msgsize = mqueueCB->mqcb->queueSize - sizeof(UINT32); mqAttr->mq_curmsgs = mqueueCB->mqcb->readWriteableCnt[OS_QUEUE_READ]; mqAttr->mq_flags = privateMqPersonal->mq_flags; (VOID)pthread_mutex_unlock(&IPC_QUEUE_MUTEX); return 0; } int OsMqSetAttr(mqd_t personal, const struct mq_attr *mqSetAttr, struct mq_attr *mqOldAttr) { struct mqpersonal *privateMqPersonal = NULL; privateMqPersonal = MqGetPrivDataBuff(personal); if (privateMqPersonal == NULL) { return -1; } if (mqSetAttr == NULL) { errno = EINVAL; return -1; } (VOID)pthread_mutex_lock(&IPC_QUEUE_MUTEX); if (privateMqPersonal->mq_status != MQ_USE_MAGIC) { errno = EBADF; (VOID)pthread_mutex_unlock(&IPC_QUEUE_MUTEX); return -1; } if (mqOldAttr != NULL) { (VOID)OsMqGetAttr(personal, mqOldAttr); } privateMqPersonal->mq_flags = (INT32)((UINT32)privateMqPersonal->mq_flags & (UINT32)(~FNONBLOCK)); /* clear */ if (((UINT32)mqSetAttr->mq_flags & (UINT32)FNONBLOCK) == (UINT32)FNONBLOCK) { privateMqPersonal->mq_flags = (INT32)((UINT32)privateMqPersonal->mq_flags | (UINT32)FNONBLOCK); } (VOID)pthread_mutex_unlock(&IPC_QUEUE_MUTEX); return 0; } int mq_getsetattr(mqd_t mqd, const struct mq_attr *new, struct mq_attr *old) { if (new == NULL) { return OsMqGetAttr(mqd, old); } return OsMqSetAttr(mqd, new, old); } int mq_unlink(const char *mqName) { INT32 ret = 0; struct mqarray *mqueueCB = NULL; if (MqNameCheck(mqName) == -1) { return -1; } (VOID)pthread_mutex_lock(&IPC_QUEUE_MUTEX); mqueueCB = GetMqueueCBByName(mqName); if (mqueueCB == NULL) { errno = ENOENT; goto ERROUT_UNLOCK; } if (mqueueCB->mq_personal != NULL) { mqueueCB->unlinkflag = TRUE; } else if (mqueueCB->unlink_ref == 0) { ret = DoMqueueDelete(mqueueCB); } (VOID)pthread_mutex_unlock(&IPC_QUEUE_MUTEX); return ret; ERROUT_UNLOCK: (VOID)pthread_mutex_unlock(&IPC_QUEUE_MUTEX); return -1; } STATIC INT32 ConvertTimeout(long flags, const struct timespec *absTimeout, UINT64 *ticks) { if ((UINT32)flags & (UINT32)FNONBLOCK) { *ticks = LOS_NO_WAIT; return 0; } if (absTimeout == NULL) { *ticks = LOS_WAIT_FOREVER; return 0; } if (!ValidTimeSpec(absTimeout)) { errno = EINVAL; return -1; } *ticks = OsTimeSpec2Tick(absTimeout); return 0; } STATIC INLINE BOOL MqParamCheck(mqd_t personal, const char *msg, size_t msgLen) { if (personal < 0) { return FALSE; } if ((msg == NULL) || (msgLen == 0)) { errno = EINVAL; return FALSE; } return TRUE; } /* * Send realtime a signal to process which registered itself * successfully by mq_notify. */ static void MqSendNotify(struct mqarray *mqueueCB) { struct mqnotify *mqnotify = &mqueueCB->mq_notify; if ((mqnotify->pid) && (mqueueCB->mqcb->readWriteableCnt[OS_QUEUE_READ] == 0)) { siginfo_t info; switch (mqnotify->notify.sigev_notify) { case SIGEV_SIGNAL: /* sends signal */ /* Create the siginfo structure */ info.si_signo = mqnotify->notify.sigev_signo; info.si_code = SI_MESGQ; info.si_value = mqnotify->notify.sigev_value; OsDispatch(mqnotify->pid, &info, OS_USER_KILL_PERMISSION); break; case SIGEV_NONE: default: break; } /* after notification unregisters process */ mqnotify->pid = 0; } } #define OS_MQ_GOTO_ERROUT_UNLOCK_IF(expr, errcode) \ if (expr) { \ errno = errcode; \ goto ERROUT_UNLOCK; \ } #define OS_MQ_GOTO_ERROUT_IF(expr, errcode) \ if (expr) { \ errno = errcode; \ goto ERROUT; \ } int mq_timedsend(mqd_t personal, const char *msg, size_t msgLen, unsigned int msgPrio, const struct timespec *absTimeout) { UINT32 mqueueID, err; UINT64 absTicks; struct mqarray *mqueueCB = NULL; struct mqpersonal *privateMqPersonal = NULL; OS_MQ_GOTO_ERROUT_IF(!MqParamCheck(personal, msg, msgLen), errno); OS_MQ_GOTO_ERROUT_IF(msgPrio > (MQ_PRIO_MAX - 1), EINVAL); (VOID)pthread_mutex_lock(&IPC_QUEUE_MUTEX); privateMqPersonal = MqGetPrivDataBuff(personal); OS_MQ_GOTO_ERROUT_UNLOCK_IF(privateMqPersonal == NULL || privateMqPersonal->mq_status != MQ_USE_MAGIC, EBADF); mqueueCB = privateMqPersonal->mq_posixdes; OS_MQ_GOTO_ERROUT_UNLOCK_IF(msgLen > (size_t)(mqueueCB->mqcb->queueSize - sizeof(UINT32)), EMSGSIZE); OS_MQ_GOTO_ERROUT_UNLOCK_IF((((UINT32)privateMqPersonal->mq_flags & (UINT32)O_WRONLY) != (UINT32)O_WRONLY) && (((UINT32)privateMqPersonal->mq_flags & (UINT32)O_RDWR) != (UINT32)O_RDWR), EBADF); OS_MQ_GOTO_ERROUT_UNLOCK_IF(ConvertTimeout(privateMqPersonal->mq_flags, absTimeout, &absTicks) == -1, errno); mqueueID = mqueueCB->mq_id; (VOID)pthread_mutex_unlock(&IPC_QUEUE_MUTEX); if (LOS_ListEmpty(&mqueueCB->mqcb->readWriteList[OS_QUEUE_READ])) { MqSendNotify(mqueueCB); } err = LOS_QueueWriteCopy(mqueueID, (VOID *)msg, (UINT32)msgLen, (UINT32)absTicks); if (map_errno(err) != ENOERR) { goto ERROUT; } return 0; ERROUT_UNLOCK: (VOID)pthread_mutex_unlock(&IPC_QUEUE_MUTEX); ERROUT: return -1; } ssize_t mq_timedreceive(mqd_t personal, char *msg, size_t msgLen, unsigned int *msgPrio, const struct timespec *absTimeout) { UINT32 mqueueID, err; UINT32 receiveLen; UINT64 absTicks; struct mqarray *mqueueCB = NULL; struct mqpersonal *privateMqPersonal = NULL; if (!MqParamCheck(personal, msg, msgLen)) { goto ERROUT; } if (msgPrio != NULL) { *msgPrio = 0; } (VOID)pthread_mutex_lock(&IPC_QUEUE_MUTEX); privateMqPersonal = MqGetPrivDataBuff(personal); if (privateMqPersonal == NULL || privateMqPersonal->mq_status != MQ_USE_MAGIC) { errno = EBADF; goto ERROUT_UNLOCK; } mqueueCB = privateMqPersonal->mq_posixdes; if (msgLen < (size_t)(mqueueCB->mqcb->queueSize - sizeof(UINT32))) { errno = EMSGSIZE; goto ERROUT_UNLOCK; } if (((UINT32)privateMqPersonal->mq_flags & (UINT32)O_WRONLY) == (UINT32)O_WRONLY) { errno = EBADF; goto ERROUT_UNLOCK; } if (ConvertTimeout(privateMqPersonal->mq_flags, absTimeout, &absTicks) == -1) { goto ERROUT_UNLOCK; } receiveLen = msgLen; mqueueID = mqueueCB->mq_id; (VOID)pthread_mutex_unlock(&IPC_QUEUE_MUTEX); err = LOS_QueueReadCopy(mqueueID, (VOID *)msg, &receiveLen, (UINT32)absTicks); if (map_errno(err) == ENOERR) { return (ssize_t)receiveLen; } else { goto ERROUT; } ERROUT_UNLOCK: (VOID)pthread_mutex_unlock(&IPC_QUEUE_MUTEX); ERROUT: return -1; } /* not support the prio */ int mq_send(mqd_t personal, const char *msg_ptr, size_t msg_len, unsigned int msg_prio) { return mq_timedsend(personal, msg_ptr, msg_len, msg_prio, NULL); } ssize_t mq_receive(mqd_t personal, char *msg_ptr, size_t msg_len, unsigned int *msg_prio) { return mq_timedreceive(personal, msg_ptr, msg_len, msg_prio, NULL); } STATIC INLINE BOOL MqNotifyParamCheck(mqd_t personal, const struct sigevent *sigev) { if (personal < 0) { errno = EBADF; goto ERROUT; } if (sigev != NULL) { if (sigev->sigev_notify != SIGEV_NONE && sigev->sigev_notify != SIGEV_SIGNAL) { errno = EINVAL; goto ERROUT; } if (sigev->sigev_notify == SIGEV_SIGNAL && !GOOD_SIGNO(sigev->sigev_signo)) { errno = EINVAL; goto ERROUT; } } return TRUE; ERROUT: return FALSE; } int OsMqNotify(mqd_t personal, const struct sigevent *sigev) { struct mqarray *mqueueCB = NULL; struct mqnotify *mqnotify = NULL; struct mqpersonal *privateMqPersonal = NULL; if (!MqNotifyParamCheck(personal, sigev)) { goto ERROUT; } (VOID)pthread_mutex_lock(&IPC_QUEUE_MUTEX); privateMqPersonal = MqGetPrivDataBuff(personal); if (privateMqPersonal == NULL) { goto OUT_UNLOCK; } if (privateMqPersonal->mq_status != MQ_USE_MAGIC) { errno = EBADF; goto OUT_UNLOCK; } mqueueCB = privateMqPersonal->mq_posixdes; mqnotify = &mqueueCB->mq_notify; if (sigev == NULL) { if (mqnotify->pid == LOS_GetCurrProcessID()) { mqnotify->pid = 0; } } else if (mqnotify->pid != 0) { errno = EBUSY; goto OUT_UNLOCK; } else { switch (sigev->sigev_notify) { case SIGEV_NONE: mqnotify->notify.sigev_notify = SIGEV_NONE; break; case SIGEV_SIGNAL: mqnotify->notify.sigev_signo = sigev->sigev_signo; mqnotify->notify.sigev_value = sigev->sigev_value; mqnotify->notify.sigev_notify = SIGEV_SIGNAL; break; default: break; } mqnotify->pid = LOS_GetCurrProcessID(); } (VOID)pthread_mutex_unlock(&IPC_QUEUE_MUTEX); return 0; OUT_UNLOCK: (VOID)pthread_mutex_unlock(&IPC_QUEUE_MUTEX); ERROUT: return -1; } #endif