/* * Copyright (c) 2013-2019 Huawei Technologies Co., Ltd. All rights reserved. * Copyright (c) 2020-2021 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 "console.h" #include "fcntl.h" #include "sys/ioctl.h" #ifdef LOSCFG_FILE_MODE #include "stdarg.h" #endif #include "unistd.h" #include "securec.h" #ifdef LOSCFG_SHELL_DMESG #include "dmesg_pri.h" #endif #ifdef LOSCFG_SHELL #include "shcmd.h" #include "shell_pri.h" #endif #include "los_exc_pri.h" #include "los_process_pri.h" #include "los_sched_pri.h" #include "fs/path_cache.h" #include "fs/vfs_util.h" #include "user_copy.h" #include "fs/vnode.h" #define EACH_CHAR 1 #define UART_IOC_MAGIC 'u' #define UART_CFG_ATTR _IOW(UART_IOC_MAGIC, 5, int) #define UART_CFG_PRIVATE _IOW(UART_IOC_MAGIC, 6, int) /* Inter-module variable */ extern UINT32 g_uart_fputc_en; STATIC UINT32 ConsoleSendTask(UINTPTR param); STATIC UINT8 g_taskConsoleIDArray[LOSCFG_BASE_CORE_TSK_LIMIT]; STATIC SPIN_LOCK_INIT(g_consoleSpin); #define SHELL_ENTRYID_INVALID 0xFFFFFFFF #define SHELL_TASK_PRIORITY 9 #define CONSOLE_CIRBUF_EVENT 0x02U #define CONSOLE_SEND_TASK_EXIT 0x04U #define CONSOLE_SEND_TASK_RUNNING 0x10U CONSOLE_CB *g_console[CONSOLE_NUM]; #define MIN(a, b) ((a) < (b) ? (a) : (b)) /* * acquire uart driver function and filep of /dev/console, * then store uart driver function in *filepOps * and store filep of /dev/console in *privFilep. */ INT32 GetFilepOps(const struct file *filep, struct file **privFilep, const struct file_operations_vfs **filepOps) { INT32 ret; if ((filep == NULL) || (filep->f_vnode == NULL) || (filep->f_vnode->data == NULL)) { ret = EINVAL; goto ERROUT; } /* to find console device's filep(now it is *privFilep) throught i_private */ struct drv_data *drv = (struct drv_data *)filep->f_vnode->data; *privFilep = (struct file *)drv->priv; if (((*privFilep)->f_vnode == NULL) || ((*privFilep)->f_vnode->data == NULL)) { ret = EINVAL; goto ERROUT; } /* to find uart driver operation function throutht u.i_opss */ drv = (struct drv_data *)(*privFilep)->f_vnode->data; *filepOps = (const struct file_operations_vfs *)drv->ops; return ENOERR; ERROUT: set_errno(ret); return VFS_ERROR; } INT32 ConsoleTcGetAttr(INT32 fd, struct termios *termios) { struct file *filep = NULL; CONSOLE_CB *consoleCB = NULL; if ((fd >= STDIN_FILENO) && (fd <= STDERR_FILENO)) { fd = ConsoleUpdateFd(); if (fd < STDIN_FILENO) { return -EBADF; } } int ret = fs_getfilep(fd, &filep); if (ret < 0) { return -EPERM; } consoleCB = (CONSOLE_CB *)filep->f_priv; if (consoleCB == NULL) { return -EFAULT; } termios->c_lflag = consoleCB->consoleTermios.c_lflag; return LOS_OK; } INT32 ConsoleTcSetAttr(INT32 fd, INT32 actions, const struct termios *termios) { struct file *filep = NULL; CONSOLE_CB *consoleCB = NULL; (VOID)actions; if ((fd >= STDIN_FILENO) && (fd <= STDERR_FILENO)) { fd = ConsoleUpdateFd(); if (fd < STDIN_FILENO) { return -EBADF; } } int ret = fs_getfilep(fd, &filep); if (ret < 0) { return -EPERM; } consoleCB = (CONSOLE_CB *)filep->f_priv; if (consoleCB == NULL) { return -EFAULT; } consoleCB->consoleTermios.c_lflag = termios->c_lflag; return LOS_OK; } STATIC UINT32 ConsoleRefcountGet(const CONSOLE_CB *consoleCB) { return consoleCB->refCount; } STATIC VOID ConsoleRefcountSet(CONSOLE_CB *consoleCB, BOOL flag) { if (flag == TRUE) { ++(consoleCB->refCount); } else { --(consoleCB->refCount); } } BOOL IsConsoleOccupied(const CONSOLE_CB *consoleCB) { if (ConsoleRefcountGet(consoleCB) != FALSE) { return TRUE; } else { return FALSE; } } STATIC INT32 ConsoleCtrlCaptureLine(CONSOLE_CB *consoleCB) { struct termios *consoleTermios = NULL; UINT32 intSave; LOS_SpinLockSave(&g_consoleSpin, &intSave); consoleTermios = &consoleCB->consoleTermios; (VOID)ConsoleTcGetAttr(consoleCB->fd, consoleTermios); consoleTermios->c_lflag |= ICANON | ECHO; (VOID)ConsoleTcSetAttr(consoleCB->fd, 0, consoleTermios); LOS_SpinUnlockRestore(&g_consoleSpin, intSave); return LOS_OK; } STATIC INT32 ConsoleCtrlCaptureChar(CONSOLE_CB *consoleCB) { struct termios *consoleTermios = NULL; UINT32 intSave; LOS_SpinLockSave(&g_consoleSpin, &intSave); consoleTermios = &consoleCB->consoleTermios; (VOID)ConsoleTcGetAttr(consoleCB->fd, consoleTermios); consoleTermios->c_lflag &= ~(ICANON | ECHO); (VOID)ConsoleTcSetAttr(consoleCB->fd, 0, consoleTermios); LOS_SpinUnlockRestore(&g_consoleSpin, intSave); return LOS_OK; } STATIC INT32 ConsoleCtrlRightsCapture(CONSOLE_CB *consoleCB) { (VOID)LOS_SemPend(consoleCB->consoleSem, LOS_WAIT_FOREVER); if ((ConsoleRefcountGet(consoleCB) == 0) && (OsCurrTaskGet()->taskID != consoleCB->shellEntryId)) { /* not 0:indicate that shellentry is in uart_read, suspend shellentry task directly */ (VOID)LOS_TaskSuspend(consoleCB->shellEntryId); } ConsoleRefcountSet(consoleCB, TRUE); return LOS_OK; } STATIC INT32 ConsoleCtrlRightsRelease(CONSOLE_CB *consoleCB) { if (ConsoleRefcountGet(consoleCB) == 0) { PRINT_ERR("console is free\n"); (VOID)LOS_SemPost(consoleCB->consoleSem); return LOS_NOK; } else { ConsoleRefcountSet(consoleCB, FALSE); if ((ConsoleRefcountGet(consoleCB) == 0) && (OsCurrTaskGet()->taskID != consoleCB->shellEntryId)) { (VOID)LOS_TaskResume(consoleCB->shellEntryId); } } (VOID)LOS_SemPost(consoleCB->consoleSem); return LOS_OK; } STATIC CONSOLE_CB *OsGetConsoleByDevice(const CHAR *deviceName) { INT32 ret; struct Vnode *vnode = NULL; VnodeHold(); ret = VnodeLookup(deviceName, &vnode, 0); VnodeDrop(); if (ret < 0) { set_errno(EACCES); return NULL; } if (g_console[CONSOLE_SERIAL - 1]->devVnode == vnode) { return g_console[CONSOLE_SERIAL - 1]; } else if (g_console[CONSOLE_TELNET - 1]->devVnode == vnode) { return g_console[CONSOLE_TELNET - 1]; } else { set_errno(ENOENT); return NULL; } } STATIC INT32 OsGetConsoleID(const CHAR *deviceName) { if ((deviceName != NULL) && (strlen(deviceName) == strlen(SERIAL)) && (!strncmp(deviceName, SERIAL, strlen(SERIAL)))) { return CONSOLE_SERIAL; } #ifdef LOSCFG_NET_TELNET else if ((deviceName != NULL) && (strlen(deviceName) == strlen(TELNET)) && (!strncmp(deviceName, TELNET, strlen(TELNET)))) { return CONSOLE_TELNET; } #endif return -1; } STATIC INT32 OsConsoleFullpathToID(const CHAR *fullpath) { #define CONSOLE_SERIAL_1 "/dev/console1" #define CONSOLE_TELNET_2 "/dev/console2" size_t len; if (fullpath == NULL) { return -1; } len = strlen(fullpath); if ((len == strlen(CONSOLE_SERIAL_1)) && (!strncmp(fullpath, CONSOLE_SERIAL_1, strlen(CONSOLE_SERIAL_1)))) { return CONSOLE_SERIAL; } #ifdef LOSCFG_NET_TELNET else if ((len == strlen(CONSOLE_TELNET_2)) && (!strncmp(fullpath, CONSOLE_TELNET_2, strlen(CONSOLE_TELNET_2)))) { return CONSOLE_TELNET; } #endif return -1; } STATIC BOOL ConsoleFifoEmpty(const CONSOLE_CB *console) { if (console->fifoOut == console->fifoIn) { return TRUE; } return FALSE; } STATIC VOID ConsoleFifoClearup(CONSOLE_CB *console) { console->fifoOut = 0; console->fifoIn = 0; (VOID)memset_s(console->fifo, CONSOLE_FIFO_SIZE, 0, CONSOLE_FIFO_SIZE); } STATIC VOID ConsoleFifoLenUpdate(CONSOLE_CB *console) { console->currentLen = console->fifoIn - console->fifoOut; } STATIC INT32 ConsoleReadFifo(CHAR *buffer, CONSOLE_CB *console, size_t bufLen) { INT32 ret; UINT32 readNum; readNum = MIN(bufLen, console->currentLen); ret = memcpy_s(buffer, bufLen, console->fifo + console->fifoOut, readNum); if (ret != EOK) { PRINTK("%s,%d memcpy_s failed\n", __FUNCTION__, __LINE__); return -1; } console->fifoOut += readNum; if (ConsoleFifoEmpty(console)) { ConsoleFifoClearup(console); } ConsoleFifoLenUpdate(console); return (INT32)readNum; } INT32 FilepOpen(struct file *filep, const struct file_operations_vfs *fops) { INT32 ret; if (fops->open == NULL) { return -EFAULT; } /* * adopt uart open function to open filep (filep is * corresponding to filep of /dev/console) */ ret = fops->open(filep); if (ret < 0) { return -EPERM; } return ret; } STATIC INLINE VOID UserEndOfRead(CONSOLE_CB *consoleCB, struct file *filep, const struct file_operations_vfs *fops) { CHAR ch; if (consoleCB->consoleTermios.c_lflag & ECHO) { ch = '\r'; (VOID)fops->write(filep, &ch, 1); } consoleCB->fifo[consoleCB->fifoIn++] = '\n'; consoleCB->fifo[consoleCB->fifoIn] = '\0'; consoleCB->currentLen = consoleCB->fifoIn; } enum { STAT_NOMAL_KEY, STAT_ESC_KEY, STAT_MULTI_KEY }; STATIC INT32 UserShellCheckUDRL(const CHAR ch, INT32 *lastTokenType) { INT32 ret = LOS_OK; if (ch == 0x1b) { /* 0x1b: ESC */ *lastTokenType = STAT_ESC_KEY; return ret; } else if (ch == 0x5b) { /* 0x5b: first Key combination */ if (*lastTokenType == STAT_ESC_KEY) { *lastTokenType = STAT_MULTI_KEY; return ret; } } else if (ch == 0x41) { /* up */ if (*lastTokenType == STAT_MULTI_KEY) { *lastTokenType = STAT_NOMAL_KEY; return ret; } } else if (ch == 0x42) { /* down */ if (*lastTokenType == STAT_MULTI_KEY) { *lastTokenType = STAT_NOMAL_KEY; return ret; } } else if (ch == 0x43) { /* right */ if (*lastTokenType == STAT_MULTI_KEY) { *lastTokenType = STAT_NOMAL_KEY; return ret; } } else if (ch == 0x44) { /* left */ if (*lastTokenType == STAT_MULTI_KEY) { *lastTokenType = STAT_NOMAL_KEY; return ret; } } return LOS_NOK; } STATIC INT32 IsNeedContinue(CONSOLE_CB *consoleCB, char ch, INT32 *lastTokenType) { if (((ch == '\b') && (consoleCB->consoleTermios.c_lflag & ECHO) && (ConsoleFifoEmpty(consoleCB))) || (UserShellCheckUDRL(ch, lastTokenType) == LOS_OK)) { /* parse the up/down/right/left key */ return LOS_NOK; } return LOS_OK; } STATIC VOID EchoToTerminal(CONSOLE_CB *consoleCB, struct file *filep, const struct file_operations_vfs *fops, char ch) { if (consoleCB->consoleTermios.c_lflag & ECHO) { if (ch == '\b') { (VOID)fops->write(filep, "\b \b", 3); } else { (VOID)fops->write(filep, &ch, EACH_CHAR); } } } STATIC VOID StoreReadChar(CONSOLE_CB *consoleCB, char ch, INT32 readcount) { if ((readcount == EACH_CHAR) && (consoleCB->fifoIn <= (CONSOLE_FIFO_SIZE - 3))) { if (ch == '\b') { if (!ConsoleFifoEmpty(consoleCB)) { consoleCB->fifo[--consoleCB->fifoIn] = '\0'; } } else { consoleCB->fifo[consoleCB->fifoIn] = (UINT8)ch; consoleCB->fifoIn++; } } } STATIC INT32 UserFilepRead(CONSOLE_CB *consoleCB, struct file *filep, const struct file_operations_vfs *fops, CHAR *buffer, size_t bufLen) { INT32 ret; INT32 needreturn = LOS_NOK; CHAR ch; INT32 lastTokenType = STAT_NOMAL_KEY; if (fops->read == NULL) { return -EFAULT; } /* Non-ICANON mode */ if ((consoleCB->consoleTermios.c_lflag & ICANON) == 0) { ret = fops->read(filep, buffer, bufLen); if (ret < 0) { return -EPERM; } return ret; } /* ICANON mode: store data to console buffer, read data and stored data into console fifo */ if (consoleCB->currentLen == 0) { while (1) { ret = fops->read(filep, &ch, EACH_CHAR); if (ret <= 0) { return ret; } if (IsNeedContinue(consoleCB, ch, &lastTokenType)) continue; switch (ch) { case '\r': ch = '\n'; case '\n': EchoToTerminal(consoleCB, filep, fops, ch); UserEndOfRead(consoleCB, filep, fops); ret = ConsoleReadFifo(buffer, consoleCB, bufLen); needreturn = LOS_OK; break; case '\b': default: EchoToTerminal(consoleCB, filep, fops, ch); StoreReadChar(consoleCB, ch, ret); break; } if (needreturn == LOS_OK) break; } } else { /* if data is already in console fifo, we returen them immediately */ ret = ConsoleReadFifo(buffer, consoleCB, bufLen); } return ret; } INT32 FilepRead(struct file *filep, const struct file_operations_vfs *fops, CHAR *buffer, size_t bufLen) { INT32 ret; if (fops->read == NULL) { return -EFAULT; } /* * adopt uart read function to read data from filep * and write data to buffer (filep is * corresponding to filep of /dev/console) */ ret = fops->read(filep, buffer, bufLen); if (ret < 0) { return -EPERM; } return ret; } INT32 FilepWrite(struct file *filep, const struct file_operations_vfs *fops, const CHAR *buffer, size_t bufLen) { INT32 ret; if (fops->write == NULL) { return -EFAULT; } ret = fops->write(filep, buffer, bufLen); if (ret < 0) { return -EPERM; } return ret; } INT32 FilepClose(struct file *filep, const struct file_operations_vfs *fops) { INT32 ret; if ((fops == NULL) || (fops->close == NULL)) { return -EFAULT; } /* * adopt uart close function to open filep (filep is * corresponding to filep of /dev/console) */ ret = fops->close(filep); if (ret < 0) { return -EPERM; } return ret; } INT32 FilepIoctl(struct file *filep, const struct file_operations_vfs *fops, INT32 cmd, unsigned long arg) { INT32 ret; if (fops->ioctl == NULL) { return -EFAULT; } ret = fops->ioctl(filep, cmd, arg); if (ret < 0) { return -EPERM; } return ret; } INT32 FilepPoll(struct file *filep, const struct file_operations_vfs *fops, poll_table *fds) { INT32 ret; if (fops->poll == NULL) { return -EFAULT; } /* * adopt uart poll function to poll filep (filep is * corresponding to filep of /dev/serial) */ ret = fops->poll(filep, fds); if (ret < 0) { return -EPERM; } return ret; } STATIC INT32 ConsoleOpen(struct file *filep) { INT32 ret; UINT32 consoleID; struct file *privFilep = NULL; const struct file_operations_vfs *fileOps = NULL; consoleID = (UINT32)OsConsoleFullpathToID(filep->f_path); if (consoleID == (UINT32)-1) { ret = EPERM; goto ERROUT; } filep->f_priv = g_console[consoleID - 1]; ret = GetFilepOps(filep, &privFilep, &fileOps); if (ret != ENOERR) { ret = EINVAL; goto ERROUT; } ret = FilepOpen(privFilep, fileOps); if (ret < 0) { ret = EPERM; goto ERROUT; } return ENOERR; ERROUT: set_errno(ret); return VFS_ERROR; } STATIC INT32 ConsoleClose(struct file *filep) { INT32 ret; struct file *privFilep = NULL; const struct file_operations_vfs *fileOps = NULL; ret = GetFilepOps(filep, &privFilep, &fileOps); if (ret != ENOERR) { ret = EINVAL; goto ERROUT; } ret = FilepClose(privFilep, fileOps); if (ret < 0) { ret = EPERM; goto ERROUT; } return ENOERR; ERROUT: set_errno(ret); return VFS_ERROR; } STATIC ssize_t DoRead(CONSOLE_CB *consoleCB, CHAR *buffer, size_t bufLen, struct file *privFilep, const struct file_operations_vfs *fileOps) { INT32 ret; #ifdef LOSCFG_SHELL if (OsCurrTaskGet()->taskID == consoleCB->shellEntryId) { ret = FilepRead(privFilep, fileOps, buffer, bufLen); } else { #endif (VOID)ConsoleCtrlRightsCapture(consoleCB); ret = UserFilepRead(consoleCB, privFilep, fileOps, buffer, bufLen); (VOID)ConsoleCtrlRightsRelease(consoleCB); #ifdef LOSCFG_SHELL } #endif return ret; } STATIC ssize_t ConsoleRead(struct file *filep, CHAR *buffer, size_t bufLen) { INT32 ret; struct file *privFilep = NULL; CONSOLE_CB *consoleCB = NULL; CHAR *sbuffer = NULL; BOOL userBuf = FALSE; const struct file_operations_vfs *fileOps = NULL; if ((buffer == NULL) || (bufLen == 0)) { ret = EINVAL; goto ERROUT; } if (bufLen > CONSOLE_FIFO_SIZE) { bufLen = CONSOLE_FIFO_SIZE; } userBuf = LOS_IsUserAddressRange((vaddr_t)(UINTPTR)buffer, bufLen); ret = GetFilepOps(filep, &privFilep, &fileOps); if (ret != ENOERR) { ret = -EINVAL; goto ERROUT; } consoleCB = (CONSOLE_CB *)filep->f_priv; if (consoleCB == NULL) { consoleCB = OsGetConsoleByTaskID(OsCurrTaskGet()->taskID); if (consoleCB == NULL) { return -EFAULT; } } /* * shell task use FilepRead function to get data, * user task use UserFilepRead to get data */ sbuffer = userBuf ? LOS_MemAlloc(m_aucSysMem0, bufLen) : buffer; if (sbuffer == NULL) { ret = -ENOMEM; goto ERROUT; } ret = DoRead(consoleCB, sbuffer, bufLen, privFilep, fileOps); if (ret < 0) { goto ERROUT; } if (userBuf) { if (LOS_ArchCopyToUser(buffer, sbuffer, bufLen) != 0) { ret = -EFAULT; goto ERROUT; } LOS_MemFree(m_aucSysMem0, sbuffer); } return ret; ERROUT: if ((userBuf) && (sbuffer != NULL)) { LOS_MemFree(m_aucSysMem0, sbuffer); } set_errno(-ret); return VFS_ERROR; } STATIC ssize_t DoWrite(CirBufSendCB *cirBufSendCB, CHAR *buffer, size_t bufLen) { INT32 cnt; size_t writen = 0; size_t toWrite = bufLen; UINT32 intSave; #ifdef LOSCFG_SHELL_DMESG (VOID)OsLogMemcpyRecord(buffer, bufLen); if (OsCheckConsoleLock()) { return 0; } #endif LOS_CirBufLock(&cirBufSendCB->cirBufCB, &intSave); while (writen < (INT32)bufLen) { /* Transform for CR/LR mode */ if ((buffer[writen] == '\n') || (buffer[writen] == '\r')) { (VOID)LOS_CirBufWrite(&cirBufSendCB->cirBufCB, "\r", 1); } cnt = LOS_CirBufWrite(&cirBufSendCB->cirBufCB, &buffer[writen], 1); if (cnt <= 0) { break; } toWrite -= cnt; writen += cnt; } LOS_CirBufUnlock(&cirBufSendCB->cirBufCB, intSave); /* Log is cached but not printed when a system exception occurs */ if (OsGetSystemStatus() == OS_SYSTEM_NORMAL) { (VOID)LOS_EventWrite(&cirBufSendCB->sendEvent, CONSOLE_CIRBUF_EVENT); } return writen; } STATIC ssize_t ConsoleWrite(struct file *filep, const CHAR *buffer, size_t bufLen) { INT32 ret; CHAR *sbuffer = NULL; BOOL userBuf = FALSE; CirBufSendCB *cirBufSendCB = NULL; struct file *privFilep = NULL; const struct file_operations_vfs *fileOps = NULL; if ((buffer == NULL) || (bufLen == 0)) { ret = EINVAL; goto ERROUT; } if (bufLen > CONSOLE_FIFO_SIZE) { bufLen = CONSOLE_FIFO_SIZE; } userBuf = LOS_IsUserAddressRange((vaddr_t)(UINTPTR)buffer, bufLen); ret = GetFilepOps(filep, &privFilep, &fileOps); if ((ret != ENOERR) || (fileOps->write == NULL) || (filep->f_priv == NULL)) { ret = EINVAL; goto ERROUT; } cirBufSendCB = ((CONSOLE_CB *)filep->f_priv)->cirBufSendCB; /* * adopt uart open function to read data from buffer * and write data to filep (filep is * corresponding to filep of /dev/console) */ sbuffer = userBuf ? LOS_MemAlloc(m_aucSysMem0, bufLen) : (CHAR *)buffer; if (sbuffer == NULL) { ret = ENOMEM; goto ERROUT; } if (userBuf && (LOS_ArchCopyFromUser(sbuffer, buffer, bufLen) != 0)) { ret = EFAULT; goto ERROUT; } ret = DoWrite(cirBufSendCB, sbuffer, bufLen); if (userBuf) { LOS_MemFree(m_aucSysMem0, sbuffer); } return ret; ERROUT: if (userBuf && sbuffer != NULL) { LOS_MemFree(m_aucSysMem0, sbuffer); } set_errno(ret); return VFS_ERROR; } STATIC INT32 ConsoleIoctl(struct file *filep, INT32 cmd, unsigned long arg) { INT32 ret; struct file *privFilep = NULL; CONSOLE_CB *consoleCB = NULL; const struct file_operations_vfs *fileOps = NULL; ret = GetFilepOps(filep, &privFilep, &fileOps); if (ret != ENOERR) { ret = EINVAL; goto ERROUT; } if (fileOps->ioctl == NULL) { ret = EFAULT; goto ERROUT; } consoleCB = (CONSOLE_CB *)filep->f_priv; if (consoleCB == NULL) { ret = EINVAL; goto ERROUT; } switch (cmd) { case CONSOLE_CONTROL_RIGHTS_CAPTURE: ret = ConsoleCtrlRightsCapture(consoleCB); break; case CONSOLE_CONTROL_RIGHTS_RELEASE: ret = ConsoleCtrlRightsRelease(consoleCB); break; case CONSOLE_CONTROL_CAPTURE_LINE: ret = ConsoleCtrlCaptureLine(consoleCB); break; case CONSOLE_CONTROL_CAPTURE_CHAR: ret = ConsoleCtrlCaptureChar(consoleCB); break; case CONSOLE_CONTROL_REG_USERTASK: ret = ConsoleTaskReg(consoleCB->consoleID, arg); break; default: if ((cmd == UART_CFG_ATTR || cmd == UART_CFG_PRIVATE) && !LOS_IsUserAddress(arg)) { ret = EINVAL; goto ERROUT; } ret = fileOps->ioctl(privFilep, cmd, arg); break; } if (ret < 0) { ret = EPERM; goto ERROUT; } return ret; ERROUT: set_errno(ret); return VFS_ERROR; } STATIC INT32 ConsolePoll(struct file *filep, poll_table *fds) { INT32 ret; struct file *privFilep = NULL; const struct file_operations_vfs *fileOps = NULL; ret = GetFilepOps(filep, &privFilep, &fileOps); if (ret != ENOERR) { ret = EINVAL; goto ERROUT; } ret = FilepPoll(privFilep, fileOps, fds); if (ret < 0) { ret = EPERM; goto ERROUT; } return ret; ERROUT: set_errno(ret); return VFS_ERROR; } /* console device driver function structure */ STATIC const struct file_operations_vfs g_consoleDevOps = { .open = ConsoleOpen, /* open */ .close = ConsoleClose, /* close */ .read = ConsoleRead, /* read */ .write = ConsoleWrite, /* write */ .seek = NULL, .ioctl = ConsoleIoctl, .mmap = NULL, #ifndef CONFIG_DISABLE_POLL .poll = ConsolePoll, #endif }; STATIC VOID OsConsoleTermiosInit(CONSOLE_CB *consoleCB, const CHAR *deviceName) { struct termios consoleTermios; if ((deviceName != NULL) && (strlen(deviceName) == strlen(SERIAL)) && (!strncmp(deviceName, SERIAL, strlen(SERIAL)))) { consoleCB->isNonBlock = SetSerialBlock(consoleCB); /* set console to have a buffer for user */ (VOID)ConsoleTcGetAttr(consoleCB->fd, &consoleTermios); consoleTermios.c_lflag |= ICANON | ECHO; (VOID)ConsoleTcSetAttr(consoleCB->fd, 0, &consoleTermios); } #ifdef LOSCFG_NET_TELNET else if ((deviceName != NULL) && (strlen(deviceName) == strlen(TELNET)) && (!strncmp(deviceName, TELNET, strlen(TELNET)))) { consoleCB->isNonBlock = SetTelnetBlock(consoleCB); /* set console to have a buffer for user */ (VOID)ConsoleTcGetAttr(consoleCB->fd, &consoleTermios); consoleTermios.c_lflag |= ICANON | ECHO; (VOID)ConsoleTcSetAttr(consoleCB->fd, 0, &consoleTermios); } #endif } STATIC INT32 OsConsoleFileInit(CONSOLE_CB *consoleCB) { INT32 ret; struct Vnode *vnode = NULL; struct file *filep = NULL; CHAR *fullpath = NULL; ret = vfs_normalize_path(NULL, consoleCB->name, &fullpath); if (ret < 0) { return EINVAL; } VnodeHold(); ret = VnodeLookup(fullpath, &vnode, 0); if (ret != LOS_OK) { ret = EACCES; goto ERROUT_WITH_FULLPATH; } consoleCB->fd = files_allocate(vnode, O_RDWR, (off_t)0, consoleCB, STDERR_FILENO + 1); if (consoleCB->fd < 0) { ret = EMFILE; goto ERROUT_WITH_FULLPATH; } ret = fs_getfilep(consoleCB->fd, &filep); if (ret < 0) { ret = EPERM; goto ERROUT_WITH_FULLPATH; } filep->f_path = fullpath; filep->ops = (struct file_operations_vfs *)((struct drv_data *)vnode->data)->ops; VnodeDrop(); return LOS_OK; ERROUT_WITH_FULLPATH: VnodeDrop(); free(fullpath); return ret; } /* * Initialized console control platform so that when we operate /dev/console * as if we are operating /dev/ttyS0 (uart0). */ STATIC INT32 OsConsoleDevInit(CONSOLE_CB *consoleCB, const CHAR *deviceName) { INT32 ret; struct file *filep = NULL; struct Vnode *vnode = NULL; /* allocate memory for filep,in order to unchange the value of filep */ filep = (struct file *)LOS_MemAlloc(m_aucSysMem0, sizeof(struct file)); if (filep == NULL) { ret = ENOMEM; goto ERROUT; } VnodeHold(); ret = VnodeLookup(deviceName, &vnode, V_DUMMY); VnodeDrop(); // not correct, but can't fix perfectly here if (ret != LOS_OK) { ret = EACCES; PRINTK("!! can not find %s\n", consoleCB->name); goto ERROUT; } consoleCB->devVnode = vnode; /* * initialize the console filep which is associated with /dev/console, * assign the uart0 vnode of /dev/ttyS0 to console inod of /dev/console, * then we can operate console's filep as if we operate uart0 filep of * /dev/ttyS0. */ (VOID)memset_s(filep, sizeof(struct file), 0, sizeof(struct file)); filep->f_oflags = O_RDWR; filep->f_pos = 0; filep->f_vnode = vnode; filep->f_path = NULL; filep->f_priv = NULL; /* * Use filep to connect console and uart, we can find uart driver function throught filep. * now we can operate /dev/console to operate /dev/ttyS0 through filep. */ ret = register_driver(consoleCB->name, &g_consoleDevOps, DEFFILEMODE, filep); if (ret != LOS_OK) { goto ERROUT; } return LOS_OK; ERROUT: if (filep) { (VOID)LOS_MemFree(m_aucSysMem0, filep); } set_errno(ret); return LOS_NOK; } STATIC UINT32 OsConsoleDevDeinit(const CONSOLE_CB *consoleCB) { return unregister_driver(consoleCB->name); } STATIC CirBufSendCB *ConsoleCirBufCreate(VOID) { UINT32 ret; CHAR *fifo = NULL; CirBufSendCB *cirBufSendCB = NULL; CirBuf *cirBufCB = NULL; cirBufSendCB = (CirBufSendCB *)LOS_MemAlloc(m_aucSysMem0, sizeof(CirBufSendCB)); if (cirBufSendCB == NULL) { return NULL; } (VOID)memset_s(cirBufSendCB, sizeof(CirBufSendCB), 0, sizeof(CirBufSendCB)); fifo = (CHAR *)LOS_MemAlloc(m_aucSysMem0, CONSOLE_CIRCBUF_SIZE); if (fifo == NULL) { goto ERROR_WITH_SENDCB; } (VOID)memset_s(fifo, CONSOLE_CIRCBUF_SIZE, 0, CONSOLE_CIRCBUF_SIZE); cirBufCB = &cirBufSendCB->cirBufCB; ret = LOS_CirBufInit(cirBufCB, fifo, CONSOLE_CIRCBUF_SIZE); if (ret != LOS_OK) { goto ERROR_WITH_FIFO; } (VOID)LOS_EventInit(&cirBufSendCB->sendEvent); return cirBufSendCB; ERROR_WITH_FIFO: (VOID)LOS_MemFree(m_aucSysMem0, cirBufCB->fifo); ERROR_WITH_SENDCB: (VOID)LOS_MemFree(m_aucSysMem0, cirBufSendCB); return NULL; } STATIC VOID ConsoleCirBufDelete(CirBufSendCB *cirBufSendCB) { CirBuf *cirBufCB = &cirBufSendCB->cirBufCB; (VOID)LOS_MemFree(m_aucSysMem0, cirBufCB->fifo); LOS_CirBufDeinit(cirBufCB); (VOID)LOS_EventDestroy(&cirBufSendCB->sendEvent); (VOID)LOS_MemFree(m_aucSysMem0, cirBufSendCB); } STATIC UINT32 OsConsoleBufInit(CONSOLE_CB *consoleCB) { UINT32 ret; TSK_INIT_PARAM_S initParam = {0}; consoleCB->cirBufSendCB = ConsoleCirBufCreate(); if (consoleCB->cirBufSendCB == NULL) { return LOS_NOK; } initParam.pfnTaskEntry = (TSK_ENTRY_FUNC)ConsoleSendTask; initParam.usTaskPrio = SHELL_TASK_PRIORITY; initParam.auwArgs[0] = (UINTPTR)consoleCB; initParam.uwStackSize = LOSCFG_BASE_CORE_TSK_DEFAULT_STACK_SIZE; if (consoleCB->consoleID == CONSOLE_SERIAL) { initParam.pcName = "SendToSer"; } else { initParam.pcName = "SendToTelnet"; } initParam.uwResved = LOS_TASK_STATUS_DETACHED; ret = LOS_TaskCreate(&consoleCB->sendTaskID, &initParam); if (ret != LOS_OK) { ConsoleCirBufDelete(consoleCB->cirBufSendCB); consoleCB->cirBufSendCB = NULL; return LOS_NOK; } (VOID)LOS_EventRead(&consoleCB->cirBufSendCB->sendEvent, CONSOLE_SEND_TASK_RUNNING, LOS_WAITMODE_OR | LOS_WAITMODE_CLR, LOS_WAIT_FOREVER); return LOS_OK; } STATIC VOID OsConsoleBufDeinit(CONSOLE_CB *consoleCB) { CirBufSendCB *cirBufSendCB = consoleCB->cirBufSendCB; consoleCB->cirBufSendCB = NULL; (VOID)LOS_EventWrite(&cirBufSendCB->sendEvent, CONSOLE_SEND_TASK_EXIT); } STATIC CONSOLE_CB *OsConsoleCBInit(UINT32 consoleID) { CONSOLE_CB *consoleCB = (CONSOLE_CB *)LOS_MemAlloc((VOID *)m_aucSysMem0, sizeof(CONSOLE_CB)); if (consoleCB == NULL) { return NULL; } (VOID)memset_s(consoleCB, sizeof(CONSOLE_CB), 0, sizeof(CONSOLE_CB)); consoleCB->consoleID = consoleID; consoleCB->shellEntryId = SHELL_ENTRYID_INVALID; /* initialize shellEntryId to an invalid value */ consoleCB->name = LOS_MemAlloc((VOID *)m_aucSysMem0, CONSOLE_NAMELEN); if (consoleCB->name == NULL) { PRINT_ERR("consoleCB->name malloc failed\n"); (VOID)LOS_MemFree((VOID *)m_aucSysMem0, consoleCB); return NULL; } return consoleCB; } STATIC VOID OsConsoleCBDeinit(CONSOLE_CB *consoleCB) { (VOID)LOS_MemFree((VOID *)m_aucSysMem0, consoleCB->name); consoleCB->name = NULL; (VOID)LOS_MemFree((VOID *)m_aucSysMem0, consoleCB); } STATIC CONSOLE_CB *OsConsoleCreate(UINT32 consoleID, const CHAR *deviceName) { INT32 ret; CONSOLE_CB *consoleCB = OsConsoleCBInit(consoleID); if (consoleCB == NULL) { PRINT_ERR("console malloc error.\n"); return NULL; } ret = snprintf_s(consoleCB->name, CONSOLE_NAMELEN, CONSOLE_NAMELEN - 1, "%s%u", CONSOLE, consoleCB->consoleID); if (ret == -1) { PRINT_ERR("consoleCB->name snprintf_s failed\n"); goto ERR_WITH_NAME; } ret = (INT32)OsConsoleBufInit(consoleCB); if (ret != LOS_OK) { PRINT_ERR("console OsConsoleBufInit error. %d\n", ret); goto ERR_WITH_NAME; } ret = (INT32)LOS_SemCreate(1, &consoleCB->consoleSem); if (ret != LOS_OK) { PRINT_ERR("creat sem for uart failed\n"); goto ERR_WITH_BUF; } ret = OsConsoleDevInit(consoleCB, deviceName); if (ret != LOS_OK) { PRINT_ERR("console OsConsoleDevInitlloc error. %d\n", ret); goto ERR_WITH_SEM; } ret = OsConsoleFileInit(consoleCB); if (ret != LOS_OK) { PRINT_ERR("console OsConsoleFileInit error. %d\n", ret); goto ERR_WITH_DEV; } OsConsoleTermiosInit(consoleCB, deviceName); return consoleCB; ERR_WITH_DEV: ret = (INT32)OsConsoleDevDeinit(consoleCB); if (ret != LOS_OK) { PRINT_ERR("OsConsoleDevDeinit failed!\n"); } ERR_WITH_SEM: (VOID)LOS_SemDelete(consoleCB->consoleSem); ERR_WITH_BUF: OsConsoleBufDeinit(consoleCB); ERR_WITH_NAME: OsConsoleCBDeinit(consoleCB); return NULL; } STATIC UINT32 OsConsoleDelete(CONSOLE_CB *consoleCB) { UINT32 ret; (VOID)files_close(consoleCB->fd); ret = OsConsoleDevDeinit(consoleCB); if (ret != LOS_OK) { PRINT_ERR("OsConsoleDevDeinit failed!\n"); } OsConsoleBufDeinit((CONSOLE_CB *)consoleCB); (VOID)LOS_SemDelete(consoleCB->consoleSem); (VOID)LOS_MemFree(m_aucSysMem0, consoleCB->name); consoleCB->name = NULL; (VOID)LOS_MemFree(m_aucSysMem0, consoleCB); return ret; } /* Initialized system console and return stdinfd stdoutfd stderrfd */ INT32 system_console_init(const CHAR *deviceName) { #ifdef LOSCFG_SHELL UINT32 ret; #endif INT32 consoleID; UINT32 intSave; CONSOLE_CB *consoleCB = NULL; consoleID = OsGetConsoleID(deviceName); if (consoleID == -1) { PRINT_ERR("device is full.\n"); return VFS_ERROR; } consoleCB = OsConsoleCreate((UINT32)consoleID, deviceName); if (consoleCB == NULL) { PRINT_ERR("%s, %d\n", __FUNCTION__, __LINE__); return VFS_ERROR; } LOS_SpinLockSave(&g_consoleSpin, &intSave); g_console[consoleID - 1] = consoleCB; if (OsCurrTaskGet() != NULL) { g_taskConsoleIDArray[OsCurrTaskGet()->taskID] = (UINT8)consoleID; } LOS_SpinUnlockRestore(&g_consoleSpin, intSave); #ifdef LOSCFG_SHELL ret = OsShellInit(consoleID); if (ret != LOS_OK) { PRINT_ERR("%s, %d\n", __FUNCTION__, __LINE__); LOS_SpinLockSave(&g_consoleSpin, &intSave); (VOID)OsConsoleDelete(consoleCB); g_console[consoleID - 1] = NULL; g_taskConsoleIDArray[OsCurrTaskGet()->taskID] = 0; LOS_SpinUnlockRestore(&g_consoleSpin, intSave); return VFS_ERROR; } #endif return ENOERR; } INT32 system_console_deinit(const CHAR *deviceName) { UINT32 ret; CONSOLE_CB *consoleCB = NULL; UINT32 taskIdx; LosTaskCB *taskCB = NULL; UINT32 intSave; consoleCB = OsGetConsoleByDevice(deviceName); if (consoleCB == NULL) { return VFS_ERROR; } #ifdef LOSCFG_SHELL (VOID)OsShellDeinit((INT32)consoleCB->consoleID); #endif LOS_SpinLockSave(&g_consoleSpin, &intSave); /* Redirect all tasks to serial as telnet was unavailable after deinitializing */ for (taskIdx = 0; taskIdx < g_taskMaxNum; taskIdx++) { taskCB = ((LosTaskCB *)g_taskCBArray) + taskIdx; if (OsTaskIsUnused(taskCB)) { continue; } else { g_taskConsoleIDArray[taskCB->taskID] = CONSOLE_SERIAL; } } g_console[consoleCB->consoleID - 1] = NULL; LOS_SpinUnlockRestore(&g_consoleSpin, intSave); ret = OsConsoleDelete(consoleCB); if (ret != LOS_OK) { PRINT_ERR("%s, Failed to system_console_deinit\n", __FUNCTION__); return VFS_ERROR; } return ENOERR; } BOOL ConsoleEnable(VOID) { INT32 consoleID; if (OsCurrTaskGet() != NULL) { consoleID = g_taskConsoleIDArray[OsCurrTaskGet()->taskID]; if (g_uart_fputc_en == 0) { if ((g_console[CONSOLE_TELNET - 1] != NULL) && OsPreemptable()) { return TRUE; } else { return FALSE; } } if (consoleID == 0) { return FALSE; } else if ((consoleID == CONSOLE_TELNET) || (consoleID == CONSOLE_SERIAL)) { if ((OsGetSystemStatus() == OS_SYSTEM_NORMAL) && !OsPreemptable()) { return FALSE; } return TRUE; } #if defined (LOSCFG_DRIVERS_USB_SERIAL_GADGET) || defined (LOSCFG_DRIVERS_USB_ETH_SER_GADGET) else if ((SerialTypeGet() == SERIAL_TYPE_USBTTY_DEV) && (userial_mask_get() == 1)) { return TRUE; } #endif } return FALSE; } INT32 ConsoleTaskReg(INT32 consoleID, UINT32 taskID) { if (g_console[consoleID - 1]->shellEntryId == SHELL_ENTRYID_INVALID) { g_console[consoleID - 1]->shellEntryId = taskID; (VOID)OsSetCurrProcessGroupID(OsGetUserInitProcessID()); return LOS_OK; } return LOS_NOK; } BOOL SetSerialNonBlock(const CONSOLE_CB *consoleCB) { INT32 ret; if (consoleCB == NULL) { PRINT_ERR("%s: Input parameter is illegal\n", __FUNCTION__); return FALSE; } ret = ioctl(consoleCB->fd, CONSOLE_CMD_RD_BLOCK_SERIAL, CONSOLE_RD_NONBLOCK); if (ret != 0) { return FALSE; } return TRUE; } BOOL SetSerialBlock(const CONSOLE_CB *consoleCB) { INT32 ret; if (consoleCB == NULL) { PRINT_ERR("%s: Input parameter is illegal\n", __FUNCTION__); return TRUE; } ret = ioctl(consoleCB->fd, CONSOLE_CMD_RD_BLOCK_SERIAL, CONSOLE_RD_BLOCK); if (ret != 0) { return TRUE; } return FALSE; } BOOL SetTelnetNonBlock(const CONSOLE_CB *consoleCB) { INT32 ret; if (consoleCB == NULL) { PRINT_ERR("%s: Input parameter is illegal\n", __FUNCTION__); return FALSE; } ret = ioctl(consoleCB->fd, CONSOLE_CMD_RD_BLOCK_TELNET, CONSOLE_RD_NONBLOCK); if (ret != 0) { return FALSE; } return TRUE; } BOOL SetTelnetBlock(const CONSOLE_CB *consoleCB) { INT32 ret; if (consoleCB == NULL) { PRINT_ERR("%s: Input parameter is illegal\n", __FUNCTION__); return TRUE; } ret = ioctl(consoleCB->fd, CONSOLE_CMD_RD_BLOCK_TELNET, CONSOLE_RD_BLOCK); if (ret != 0) { return TRUE; } return FALSE; } BOOL is_nonblock(const CONSOLE_CB *consoleCB) { if (consoleCB == NULL) { PRINT_ERR("%s: Input parameter is illegal\n", __FUNCTION__); return FALSE; } return consoleCB->isNonBlock; } INT32 ConsoleUpdateFd(VOID) { INT32 consoleID; if (OsCurrTaskGet() != NULL) { consoleID = g_taskConsoleIDArray[(OsCurrTaskGet())->taskID]; } else { return -1; } if (g_uart_fputc_en == 0) { if (g_console[CONSOLE_TELNET - 1] != NULL) { consoleID = CONSOLE_TELNET; } else { return -1; } } else if (consoleID == 0) { if (g_console[CONSOLE_SERIAL - 1] != NULL) { consoleID = CONSOLE_SERIAL; } else if (g_console[CONSOLE_TELNET - 1] != NULL) { consoleID = CONSOLE_TELNET; } else { PRINTK("No console dev used.\n"); return -1; } } if (g_console[consoleID - 1] == NULL) { return -1; } return g_console[consoleID - 1]->fd; } CONSOLE_CB *OsGetConsoleByID(INT32 consoleID) { if (consoleID != CONSOLE_TELNET) { consoleID = CONSOLE_SERIAL; } return g_console[consoleID - 1]; } CONSOLE_CB *OsGetConsoleByTaskID(UINT32 taskID) { INT32 consoleID = g_taskConsoleIDArray[taskID]; return OsGetConsoleByID(consoleID); } VOID OsSetConsoleID(UINT32 newTaskID, UINT32 curTaskID) { if ((newTaskID >= LOSCFG_BASE_CORE_TSK_LIMIT) || (curTaskID >= LOSCFG_BASE_CORE_TSK_LIMIT)) { return; } g_taskConsoleIDArray[newTaskID] = g_taskConsoleIDArray[curTaskID]; } STATIC ssize_t WriteToTerminal(const CONSOLE_CB *consoleCB, const CHAR *buffer, size_t bufLen) { INT32 ret, fd; INT32 cnt = 0; struct file *privFilep = NULL; struct file *filep = NULL; const struct file_operations_vfs *fileOps = NULL; fd = consoleCB->fd; ret = fs_getfilep(fd, &filep); ret = GetFilepOps(filep, &privFilep, &fileOps); if ((fileOps == NULL) || (fileOps->write == NULL)) { ret = EFAULT; goto ERROUT; } (VOID)fileOps->write(privFilep, buffer, bufLen); return cnt; ERROUT: set_errno(ret); return VFS_ERROR; } STATIC UINT32 ConsoleSendTask(UINTPTR param) { CONSOLE_CB *consoleCB = (CONSOLE_CB *)param; CirBufSendCB *cirBufSendCB = consoleCB->cirBufSendCB; CirBuf *cirBufCB = &cirBufSendCB->cirBufCB; UINT32 ret, size; UINT32 intSave; CHAR *buf = NULL; (VOID)LOS_EventWrite(&cirBufSendCB->sendEvent, CONSOLE_SEND_TASK_RUNNING); while (1) { ret = LOS_EventRead(&cirBufSendCB->sendEvent, CONSOLE_CIRBUF_EVENT | CONSOLE_SEND_TASK_EXIT, LOS_WAITMODE_OR | LOS_WAITMODE_CLR, LOS_WAIT_FOREVER); if (ret == CONSOLE_CIRBUF_EVENT) { size = LOS_CirBufUsedSize(cirBufCB); if (size == 0) { continue; } buf = (CHAR *)LOS_MemAlloc(m_aucSysMem1, size + 1); if (buf == NULL) { continue; } (VOID)memset_s(buf, size + 1, 0, size + 1); LOS_CirBufLock(cirBufCB, &intSave); (VOID)LOS_CirBufRead(cirBufCB, buf, size); LOS_CirBufUnlock(cirBufCB, intSave); (VOID)WriteToTerminal(consoleCB, buf, size); (VOID)LOS_MemFree(m_aucSysMem1, buf); } else if (ret == CONSOLE_SEND_TASK_EXIT) { break; } } ConsoleCirBufDelete(cirBufSendCB); return LOS_OK; } #if (LOSCFG_KERNEL_SMP == YES) VOID OsWaitConsoleSendTaskPend(UINT32 taskID) { UINT32 i; CONSOLE_CB *console = NULL; LosTaskCB *taskCB = NULL; INT32 waitTime = 3000; /* 3000: 3 seconds */ for (i = 0; i < CONSOLE_NUM; i++) { console = g_console[i]; if (console == NULL) { continue; } if (OS_TID_CHECK_INVALID(console->sendTaskID)) { continue; } taskCB = OS_TCB_FROM_TID(console->sendTaskID); while ((waitTime > 0) && (taskCB->taskEvent == NULL) && (taskID != console->sendTaskID)) { LOS_Mdelay(1); /* 1: wait console task pend */ --waitTime; } } } VOID OsWakeConsoleSendTask(VOID) { UINT32 i; CONSOLE_CB *console = NULL; for (i = 0; i < CONSOLE_NUM; i++) { console = g_console[i]; if (console == NULL) { continue; } if (console->cirBufSendCB != NULL) { (VOID)LOS_EventWrite(&console->cirBufSendCB->sendEvent, CONSOLE_CIRBUF_EVENT); } } } #endif