10d163575Sopenharmony_ci/*
20d163575Sopenharmony_ci * Copyright (c) 2013-2019 Huawei Technologies Co., Ltd. All rights reserved.
30d163575Sopenharmony_ci * Copyright (c) 2020-2021 Huawei Device Co., Ltd. All rights reserved.
40d163575Sopenharmony_ci *
50d163575Sopenharmony_ci * Redistribution and use in source and binary forms, with or without modification,
60d163575Sopenharmony_ci * are permitted provided that the following conditions are met:
70d163575Sopenharmony_ci *
80d163575Sopenharmony_ci * 1. Redistributions of source code must retain the above copyright notice, this list of
90d163575Sopenharmony_ci *    conditions and the following disclaimer.
100d163575Sopenharmony_ci *
110d163575Sopenharmony_ci * 2. Redistributions in binary form must reproduce the above copyright notice, this list
120d163575Sopenharmony_ci *    of conditions and the following disclaimer in the documentation and/or other materials
130d163575Sopenharmony_ci *    provided with the distribution.
140d163575Sopenharmony_ci *
150d163575Sopenharmony_ci * 3. Neither the name of the copyright holder nor the names of its contributors may be used
160d163575Sopenharmony_ci *    to endorse or promote products derived from this software without specific prior written
170d163575Sopenharmony_ci *    permission.
180d163575Sopenharmony_ci *
190d163575Sopenharmony_ci * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
200d163575Sopenharmony_ci * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
210d163575Sopenharmony_ci * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
220d163575Sopenharmony_ci * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
230d163575Sopenharmony_ci * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
240d163575Sopenharmony_ci * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
250d163575Sopenharmony_ci * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
260d163575Sopenharmony_ci * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
270d163575Sopenharmony_ci * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
280d163575Sopenharmony_ci * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
290d163575Sopenharmony_ci * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
300d163575Sopenharmony_ci */
310d163575Sopenharmony_ci
320d163575Sopenharmony_ci#include "virtual_serial.h"
330d163575Sopenharmony_ci#include "fcntl.h"
340d163575Sopenharmony_ci#ifdef LOSCFG_FILE_MODE
350d163575Sopenharmony_ci#include "stdarg.h"
360d163575Sopenharmony_ci#endif
370d163575Sopenharmony_ci#ifdef LOSCFG_FS_VFS
380d163575Sopenharmony_ci#include "console.h"
390d163575Sopenharmony_ci#include "fs/driver.h"
400d163575Sopenharmony_ci#endif
410d163575Sopenharmony_ci
420d163575Sopenharmony_ciSTATIC volatile UINT32 g_serialType = 0;
430d163575Sopenharmony_ciSTATIC struct file g_serialFilep;
440d163575Sopenharmony_ci
450d163575Sopenharmony_ci
460d163575Sopenharmony_ciUINT32 SerialTypeGet(VOID)
470d163575Sopenharmony_ci{
480d163575Sopenharmony_ci    return g_serialType;
490d163575Sopenharmony_ci}
500d163575Sopenharmony_ci
510d163575Sopenharmony_ciSTATIC VOID SerialTypeSet(const CHAR *deviceName)
520d163575Sopenharmony_ci{
530d163575Sopenharmony_ci    if (!strncmp(deviceName, SERIAL_UARTDEV, strlen(SERIAL_UARTDEV))) {
540d163575Sopenharmony_ci        g_serialType = SERIAL_TYPE_UART_DEV;
550d163575Sopenharmony_ci    } else if (!strncmp(deviceName, SERIAL_TTYGS0, strlen(SERIAL_TTYGS0))) {
560d163575Sopenharmony_ci        g_serialType = SERIAL_TYPE_USBTTY_DEV;
570d163575Sopenharmony_ci    }
580d163575Sopenharmony_ci}
590d163575Sopenharmony_ci
600d163575Sopenharmony_ciSTATIC INT32 SerialOpen(struct file *filep)
610d163575Sopenharmony_ci{
620d163575Sopenharmony_ci    INT32 ret;
630d163575Sopenharmony_ci    struct file *privFilep = NULL;
640d163575Sopenharmony_ci    const struct file_operations_vfs *fileOps = NULL;
650d163575Sopenharmony_ci
660d163575Sopenharmony_ci    ret = GetFilepOps(filep, &privFilep, &fileOps);
670d163575Sopenharmony_ci    if (ret != ENOERR) {
680d163575Sopenharmony_ci        ret = EINVAL;
690d163575Sopenharmony_ci        goto ERROUT;
700d163575Sopenharmony_ci    }
710d163575Sopenharmony_ci
720d163575Sopenharmony_ci    ret = FilepOpen(privFilep, fileOps);
730d163575Sopenharmony_ci    if (ret < 0) {
740d163575Sopenharmony_ci        ret = EPERM;
750d163575Sopenharmony_ci        goto ERROUT;
760d163575Sopenharmony_ci    }
770d163575Sopenharmony_ci
780d163575Sopenharmony_ci    if (g_serialType == SERIAL_TYPE_UART_DEV) {
790d163575Sopenharmony_ci        HalIrqUnmask(NUM_HAL_INTERRUPT_UART);
800d163575Sopenharmony_ci    }
810d163575Sopenharmony_ci    return ENOERR;
820d163575Sopenharmony_ci
830d163575Sopenharmony_ciERROUT:
840d163575Sopenharmony_ci    set_errno(ret);
850d163575Sopenharmony_ci    return VFS_ERROR;
860d163575Sopenharmony_ci}
870d163575Sopenharmony_ci
880d163575Sopenharmony_ciSTATIC INT32 SerialClose(struct file *filep)
890d163575Sopenharmony_ci{
900d163575Sopenharmony_ci    (VOID)filep;
910d163575Sopenharmony_ci
920d163575Sopenharmony_ci    if (g_serialType == SERIAL_TYPE_UART_DEV) {
930d163575Sopenharmony_ci        HalIrqMask(NUM_HAL_INTERRUPT_UART);
940d163575Sopenharmony_ci    }
950d163575Sopenharmony_ci#if defined(LOSCFG_DRIVERS_USB_SERIAL_GADGET) || defined(LOSCFG_DRIVERS_USB_ETH_SER_GADGET)
960d163575Sopenharmony_ci    else if (g_serialType == SERIAL_TYPE_USBTTY_DEV) {
970d163575Sopenharmony_ci        userial_mask_set(0);
980d163575Sopenharmony_ci    }
990d163575Sopenharmony_ci#endif
1000d163575Sopenharmony_ci
1010d163575Sopenharmony_ci    return ENOERR;
1020d163575Sopenharmony_ci}
1030d163575Sopenharmony_ci
1040d163575Sopenharmony_ciSTATIC ssize_t SerialRead(struct file *filep, CHAR *buffer, size_t bufLen)
1050d163575Sopenharmony_ci{
1060d163575Sopenharmony_ci    INT32 ret;
1070d163575Sopenharmony_ci    struct file *privFilep = NULL;
1080d163575Sopenharmony_ci    const struct file_operations_vfs *fileOps = NULL;
1090d163575Sopenharmony_ci
1100d163575Sopenharmony_ci    ret = GetFilepOps(filep, &privFilep, &fileOps);
1110d163575Sopenharmony_ci    if (ret != ENOERR) {
1120d163575Sopenharmony_ci        ret = -EINVAL;
1130d163575Sopenharmony_ci        goto ERROUT;
1140d163575Sopenharmony_ci    }
1150d163575Sopenharmony_ci
1160d163575Sopenharmony_ci    ret = FilepRead(privFilep, fileOps, buffer, bufLen);
1170d163575Sopenharmony_ci    if (ret < 0) {
1180d163575Sopenharmony_ci        goto ERROUT;
1190d163575Sopenharmony_ci    }
1200d163575Sopenharmony_ci    return ret;
1210d163575Sopenharmony_ci
1220d163575Sopenharmony_ciERROUT:
1230d163575Sopenharmony_ci    set_errno(-ret);
1240d163575Sopenharmony_ci    return VFS_ERROR;
1250d163575Sopenharmony_ci}
1260d163575Sopenharmony_ci
1270d163575Sopenharmony_ci/* Note: do not add print function in this module! */
1280d163575Sopenharmony_ciSTATIC ssize_t SerialWrite(struct file *filep,  const CHAR *buffer, size_t bufLen)
1290d163575Sopenharmony_ci{
1300d163575Sopenharmony_ci    INT32 ret;
1310d163575Sopenharmony_ci    struct file *privFilep = NULL;
1320d163575Sopenharmony_ci    const struct file_operations_vfs *fileOps = NULL;
1330d163575Sopenharmony_ci
1340d163575Sopenharmony_ci    ret = GetFilepOps(filep, &privFilep, &fileOps);
1350d163575Sopenharmony_ci    if (ret != ENOERR) {
1360d163575Sopenharmony_ci        ret = -EINVAL;
1370d163575Sopenharmony_ci        goto ERROUT;
1380d163575Sopenharmony_ci    }
1390d163575Sopenharmony_ci
1400d163575Sopenharmony_ci    ret = FilepWrite(privFilep, fileOps, buffer, bufLen);
1410d163575Sopenharmony_ci    if (ret < 0) {
1420d163575Sopenharmony_ci        goto ERROUT;
1430d163575Sopenharmony_ci    }
1440d163575Sopenharmony_ci    return ret;
1450d163575Sopenharmony_ci
1460d163575Sopenharmony_ciERROUT:
1470d163575Sopenharmony_ci    set_errno(-ret);
1480d163575Sopenharmony_ci    return VFS_ERROR;
1490d163575Sopenharmony_ci}
1500d163575Sopenharmony_ci
1510d163575Sopenharmony_ciSTATIC INT32 SerialIoctl(struct file *filep, INT32 cmd, unsigned long arg)
1520d163575Sopenharmony_ci{
1530d163575Sopenharmony_ci    INT32 ret;
1540d163575Sopenharmony_ci    struct file *privFilep = NULL;
1550d163575Sopenharmony_ci    const struct file_operations_vfs *fileOps = NULL;
1560d163575Sopenharmony_ci
1570d163575Sopenharmony_ci    ret = GetFilepOps(filep, &privFilep, &fileOps);
1580d163575Sopenharmony_ci    if (ret != ENOERR) {
1590d163575Sopenharmony_ci        ret = -EINVAL;
1600d163575Sopenharmony_ci        goto ERROUT;
1610d163575Sopenharmony_ci    }
1620d163575Sopenharmony_ci
1630d163575Sopenharmony_ci    ret = FilepIoctl(privFilep, fileOps, cmd, arg);
1640d163575Sopenharmony_ci    if (ret < 0) {
1650d163575Sopenharmony_ci        goto ERROUT;
1660d163575Sopenharmony_ci    }
1670d163575Sopenharmony_ci    return ret;
1680d163575Sopenharmony_ci
1690d163575Sopenharmony_ciERROUT:
1700d163575Sopenharmony_ci    set_errno(-ret);
1710d163575Sopenharmony_ci    return VFS_ERROR;
1720d163575Sopenharmony_ci}
1730d163575Sopenharmony_ci
1740d163575Sopenharmony_ciSTATIC INT32 SerialPoll(struct file *filep, poll_table *fds)
1750d163575Sopenharmony_ci{
1760d163575Sopenharmony_ci    INT32 ret;
1770d163575Sopenharmony_ci    struct file *privFilep = NULL;
1780d163575Sopenharmony_ci    const struct file_operations_vfs *fileOps = NULL;
1790d163575Sopenharmony_ci
1800d163575Sopenharmony_ci    ret = GetFilepOps(filep, &privFilep, &fileOps);
1810d163575Sopenharmony_ci    if (ret != ENOERR) {
1820d163575Sopenharmony_ci        ret = -EINVAL;
1830d163575Sopenharmony_ci        goto ERROUT;
1840d163575Sopenharmony_ci    }
1850d163575Sopenharmony_ci    ret = FilepPoll(privFilep, fileOps, fds);
1860d163575Sopenharmony_ci    if (ret < 0) {
1870d163575Sopenharmony_ci        goto ERROUT;
1880d163575Sopenharmony_ci    }
1890d163575Sopenharmony_ci    return ret;
1900d163575Sopenharmony_ci
1910d163575Sopenharmony_ciERROUT:
1920d163575Sopenharmony_ci    set_errno(-ret);
1930d163575Sopenharmony_ci    return VFS_ERROR;
1940d163575Sopenharmony_ci}
1950d163575Sopenharmony_ci
1960d163575Sopenharmony_ciSTATIC const struct file_operations_vfs g_serialDevOps = {
1970d163575Sopenharmony_ci    SerialOpen,  /* open */
1980d163575Sopenharmony_ci    SerialClose, /* close */
1990d163575Sopenharmony_ci    SerialRead,  /* read */
2000d163575Sopenharmony_ci    SerialWrite,
2010d163575Sopenharmony_ci    NULL,
2020d163575Sopenharmony_ci    SerialIoctl,
2030d163575Sopenharmony_ci    NULL,
2040d163575Sopenharmony_ci#ifndef CONFIG_DISABLE_POLL
2050d163575Sopenharmony_ci    SerialPoll,
2060d163575Sopenharmony_ci#endif
2070d163575Sopenharmony_ci    NULL,
2080d163575Sopenharmony_ci};
2090d163575Sopenharmony_ci
2100d163575Sopenharmony_ciINT32 virtual_serial_init(const CHAR *deviceName)
2110d163575Sopenharmony_ci{
2120d163575Sopenharmony_ci    INT32 ret;
2130d163575Sopenharmony_ci    struct Vnode *vnode = NULL;
2140d163575Sopenharmony_ci
2150d163575Sopenharmony_ci    if (deviceName == NULL) {
2160d163575Sopenharmony_ci        ret = EINVAL;
2170d163575Sopenharmony_ci        goto ERROUT;
2180d163575Sopenharmony_ci    }
2190d163575Sopenharmony_ci
2200d163575Sopenharmony_ci    SerialTypeSet(deviceName);
2210d163575Sopenharmony_ci
2220d163575Sopenharmony_ci    VnodeHold();
2230d163575Sopenharmony_ci    ret = VnodeLookup(deviceName, &vnode, V_DUMMY);
2240d163575Sopenharmony_ci    if (ret != LOS_OK) {
2250d163575Sopenharmony_ci        ret = EACCES;
2260d163575Sopenharmony_ci        goto ERROUT;
2270d163575Sopenharmony_ci    }
2280d163575Sopenharmony_ci
2290d163575Sopenharmony_ci    (VOID)memset_s(&g_serialFilep, sizeof(struct file), 0, sizeof(struct file));
2300d163575Sopenharmony_ci    g_serialFilep.f_oflags = O_RDWR;
2310d163575Sopenharmony_ci    g_serialFilep.f_vnode = vnode;
2320d163575Sopenharmony_ci    g_serialFilep.ops = ((struct drv_data *)vnode->data)->ops;
2330d163575Sopenharmony_ci
2340d163575Sopenharmony_ci    if (g_serialFilep.ops->open != NULL) {
2350d163575Sopenharmony_ci        (VOID)g_serialFilep.ops->open(&g_serialFilep);
2360d163575Sopenharmony_ci    } else {
2370d163575Sopenharmony_ci        ret = EFAULT;
2380d163575Sopenharmony_ci        PRINTK("virtual_serial_init %s open is NULL\n", deviceName);
2390d163575Sopenharmony_ci        goto ERROUT;
2400d163575Sopenharmony_ci    }
2410d163575Sopenharmony_ci    (VOID)register_driver(SERIAL, &g_serialDevOps, DEFFILEMODE, &g_serialFilep);
2420d163575Sopenharmony_ci
2430d163575Sopenharmony_ci    VnodeDrop();
2440d163575Sopenharmony_ci    return ENOERR;
2450d163575Sopenharmony_ci
2460d163575Sopenharmony_ciERROUT:
2470d163575Sopenharmony_ci    VnodeDrop();
2480d163575Sopenharmony_ci    set_errno(ret);
2490d163575Sopenharmony_ci    return VFS_ERROR;
2500d163575Sopenharmony_ci}
2510d163575Sopenharmony_ci
2520d163575Sopenharmony_ciINT32 virtual_serial_deinit(VOID)
2530d163575Sopenharmony_ci{
2540d163575Sopenharmony_ci    return unregister_driver(SERIAL);
2550d163575Sopenharmony_ci}
2560d163575Sopenharmony_ci
257