openharmony_kernel_liteos_a/net/telnet/src/telnet_dev.c

410 lines
12 KiB
C

/*
* 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 "telnet_dev.h"
#ifdef LOSCFG_NET_TELNET
#include "unistd.h"
#include "stdlib.h"
#include "sys/ioctl.h"
#include "sys/types.h"
#include "pthread.h"
#include "los_printf.h"
#if (LOSCFG_BASE_CORE_SWTMR == YES)
#include "los_swtmr_pri.h"
#endif
#include "los_sched_pri.h"
#include "console.h"
#include "lwip/opt.h"
#include "lwip/sockets.h"
#include "telnet_pri.h"
#include "fs/driver.h"
/* event: there are more commands left in the FIFO to run */
#define TELNET_EVENT_MORE_CMD 0x01
#define TELNET_DEV_DRV_MODE 0666
STATIC TELNET_DEV_S g_telnetDev;
STATIC EVENT_CB_S *g_event;
STATIC struct Vnode *g_currentVnode;
STATIC INLINE TELNET_DEV_S *GetTelnetDevByFile(const struct file *file, BOOL isOpenOp)
{
struct Vnode *telnetInode = NULL;
TELNET_DEV_S *telnetDev = NULL;
if (file == NULL) {
return NULL;
}
telnetInode = file->f_vnode;
if (telnetInode == NULL) {
return NULL;
}
/*
* Check if the f_vnode is valid here for non-open ops (open is supposed to get invalid f_vnode):
* when telnet is disconnected, there still may be 'TelentShellTask' tasks trying to write
* to the file, but the file has illegal f_vnode because the file is used by others.
*/
if (!isOpenOp) {
if (telnetInode != g_currentVnode) {
return NULL;
}
}
telnetDev = (TELNET_DEV_S *)((struct drv_data*)telnetInode->data)->priv;
return telnetDev;
}
/*
* Description : When receive user's input commands, first copy commands to the FIFO of the telnet device.
* Then, notify a command resolver task (an individual shell task) to take out and run commands.
* Return : -1 --- On failure
* Non-negative integer --- length of written commands
*/
INT32 TelnetTx(const CHAR *buf, UINT32 bufLen)
{
UINT32 i;
TELNET_DEV_S *telnetDev = NULL;
TelnetLock();
telnetDev = &g_telnetDev;
if ((buf == NULL) || (telnetDev->cmdFifo == NULL)) {
TelnetUnlock();
return -1;
}
/* size limited */
if (bufLen > telnetDev->cmdFifo->fifoNum) {
bufLen = telnetDev->cmdFifo->fifoNum;
}
if (bufLen == 0) {
TelnetUnlock();
return 0;
}
/* copy commands to the fifo of the telnet device */
for (i = 0; i < bufLen; i++) {
telnetDev->cmdFifo->rxBuf[telnetDev->cmdFifo->rxIndex] = *buf;
telnetDev->cmdFifo->rxIndex++;
telnetDev->cmdFifo->rxIndex %= FIFO_MAX;
buf++;
}
telnetDev->cmdFifo->fifoNum -= bufLen;
if (telnetDev->eventPend) {
/* signal that there are some works to do */
(VOID)LOS_EventWrite(&telnetDev->eventTelnet, TELNET_EVENT_MORE_CMD);
}
/* notify the command resolver task */
notify_poll(&telnetDev->wait);
TelnetUnlock();
return (INT32)bufLen;
}
/*
* Description : When open the telnet device, init the FIFO, wait queue etc.
*/
STATIC INT32 TelnetOpen(struct file *file)
{
struct wait_queue_head *wait = NULL;
TELNET_DEV_S *telnetDev = NULL;
TelnetLock();
telnetDev = GetTelnetDevByFile(file, TRUE);
if (telnetDev == NULL) {
TelnetUnlock();
return -1;
}
if (telnetDev->cmdFifo == NULL) {
wait = &telnetDev->wait;
(VOID)LOS_EventInit(&telnetDev->eventTelnet);
g_event = &telnetDev->eventTelnet;
telnetDev->cmdFifo = (TELNTE_FIFO_S *)malloc(sizeof(TELNTE_FIFO_S));
if (telnetDev->cmdFifo == NULL) {
TelnetUnlock();
return -1;
}
(VOID)memset_s(telnetDev->cmdFifo, sizeof(TELNTE_FIFO_S), 0, sizeof(TELNTE_FIFO_S));
telnetDev->cmdFifo->fifoNum = FIFO_MAX;
LOS_ListInit(&wait->poll_queue);
}
g_currentVnode = file->f_vnode;
TelnetUnlock();
return 0;
}
/*
* Description : When close the telnet device, free the FIFO, wait queue etc.
*/
STATIC INT32 TelnetClose(struct file *file)
{
struct wait_queue_head *wait = NULL;
TELNET_DEV_S *telnetDev = NULL;
TelnetLock();
telnetDev = GetTelnetDevByFile(file, FALSE);
if (telnetDev != NULL) {
wait = &telnetDev->wait;
LOS_ListDelete(&wait->poll_queue);
free(telnetDev->cmdFifo);
telnetDev->cmdFifo = NULL;
(VOID)LOS_EventDestroy(&telnetDev->eventTelnet);
g_event = NULL;
}
g_currentVnode = NULL;
TelnetUnlock();
return 0;
}
/*
* Description : When a command resolver task trys to read the telnet device,
* this method is called, and it will take out user's commands from the FIFO to run.
* Return : -1 --- On failure
* Non-negative integer --- length of commands taken out from the FIFO of the telnet device.
*/
STATIC ssize_t TelnetRead(struct file *file, CHAR *buf, size_t bufLen)
{
UINT32 i;
TELNET_DEV_S *telnetDev = NULL;
TelnetLock();
telnetDev = GetTelnetDevByFile(file, FALSE);
if ((buf == NULL) || (telnetDev == NULL) || (telnetDev->cmdFifo == NULL)) {
TelnetUnlock();
return -1;
}
if (telnetDev->eventPend) {
TelnetUnlock();
(VOID)LOS_EventRead(g_event, TELNET_EVENT_MORE_CMD, LOS_WAITMODE_OR, LOS_WAIT_FOREVER);
TelnetLock();
}
if (bufLen > (FIFO_MAX - telnetDev->cmdFifo->fifoNum)) {
bufLen = FIFO_MAX - telnetDev->cmdFifo->fifoNum;
}
for (i = 0; i < bufLen; i++) {
*buf++ = telnetDev->cmdFifo->rxBuf[telnetDev->cmdFifo->rxOutIndex++];
if (telnetDev->cmdFifo->rxOutIndex >= FIFO_MAX) {
telnetDev->cmdFifo->rxOutIndex = 0;
}
}
telnetDev->cmdFifo->fifoNum += bufLen;
/* check if no more commands left to run */
if (telnetDev->cmdFifo->fifoNum == FIFO_MAX) {
(VOID)LOS_EventClear(&telnetDev->eventTelnet, ~TELNET_EVENT_MORE_CMD);
}
TelnetUnlock();
return (ssize_t)bufLen;
}
/*
* Description : When a command resolver task trys to write command results to the telnet device,
* just use lwIP send function to send out results.
* Return : -1 --- buffer is NULL
* Non-negative integer --- length of written data, maybe 0.
*/
STATIC ssize_t TelnetWrite(struct file *file, const CHAR *buf, const size_t bufLen)
{
INT32 ret = 0;
TELNET_DEV_S *telnetDev = NULL;
TelnetLock();
telnetDev = GetTelnetDevByFile(file, FALSE);
if ((buf == NULL) || (telnetDev == NULL) || (telnetDev->cmdFifo == NULL)) {
TelnetUnlock();
return -1;
}
if (OS_INT_ACTIVE) {
TelnetUnlock();
return ret;
}
if (!OsPreemptable()) {
TelnetUnlock();
return ret;
}
if (telnetDev->clientFd != 0) {
#if (LOSCFG_BASE_CORE_SWTMR == YES)
/* DO NOT call blocking API in software timer task */
if (((LosTaskCB*)OsCurrTaskGet())->taskEntry == (TSK_ENTRY_FUNC)OsSwtmrTask) {
TelnetUnlock();
return ret;
}
#endif
ret = send(telnetDev->clientFd, buf, bufLen, 0);
}
TelnetUnlock();
return ret;
}
STATIC INT32 TelnetIoctl(struct file *file, const INT32 cmd, unsigned long arg)
{
TELNET_DEV_S *telnetDev = NULL;
TelnetLock();
telnetDev = GetTelnetDevByFile(file, FALSE);
if (telnetDev == NULL) {
TelnetUnlock();
return -1;
}
if (cmd == CFG_TELNET_EVENT_PEND) {
if (arg == 0) {
telnetDev->eventPend = FALSE;
(VOID)LOS_EventWrite(&(telnetDev->eventTelnet), TELNET_EVENT_MORE_CMD);
(VOID)LOS_EventClear(&(telnetDev->eventTelnet), ~TELNET_EVENT_MORE_CMD);
} else {
telnetDev->eventPend = TRUE;
}
} else if (cmd == CFG_TELNET_SET_FD) {
if (arg >= (FD_SETSIZE - 1)) {
TelnetUnlock();
return -1;
}
telnetDev->clientFd = (INT32)arg;
}
TelnetUnlock();
return 0;
}
STATIC INT32 TelnetPoll(struct file *file, poll_table *table)
{
TELNET_DEV_S *telnetDev = NULL;
TelnetLock();
telnetDev = GetTelnetDevByFile(file, FALSE);
if ((telnetDev == NULL) || (telnetDev->cmdFifo == NULL)) {
TelnetUnlock();
return -1;
}
poll_wait(file, &telnetDev->wait, table);
/* check if there are some commands to run */
if (telnetDev->cmdFifo->fifoNum != FIFO_MAX) {
TelnetUnlock();
return POLLIN | POLLRDNORM;
}
TelnetUnlock();
return 0;
}
STATIC const struct file_operations_vfs g_telnetOps = {
TelnetOpen,
TelnetClose,
TelnetRead,
TelnetWrite,
NULL,
TelnetIoctl,
NULL,
#ifndef CONFIG_DISABLE_POLL
TelnetPoll,
#endif
NULL,
};
/* Once the telnet server stopped, remove the telnet device file. */
INT32 TelnetedUnregister(VOID)
{
free(g_telnetDev.cmdFifo);
g_telnetDev.cmdFifo = NULL;
(VOID)unregister_driver(TELNET);
return 0;
}
/* Once the telnet server started, setup the telnet device file. */
INT32 TelnetedRegister(VOID)
{
INT32 ret;
g_telnetDev.id = 0;
g_telnetDev.cmdFifo = NULL;
g_telnetDev.eventPend = TRUE;
ret = register_driver(TELNET, &g_telnetOps, TELNET_DEV_DRV_MODE, &g_telnetDev);
if (ret != 0) {
PRINT_ERR("Telnet register driver error.\n");
}
return ret;
}
/* When a telnet client connection established, update the output console for tasks. */
INT32 TelnetDevInit(INT32 clientFd)
{
INT32 ret;
if (clientFd < 0) {
PRINT_ERR("Invalid telnet clientFd.\n");
return -1;
}
ret = system_console_init(TELNET);
if (ret != 0) {
PRINT_ERR("Telnet console init error.\n");
return ret;
}
ret = ioctl(STDIN_FILENO, CFG_TELNET_SET_FD, clientFd);
if (ret != 0) {
PRINT_ERR("Telnet device ioctl error.\n");
(VOID)system_console_deinit(TELNET);
}
return ret;
}
/* When closing the telnet client connection, reset the output console for tasks. */
INT32 TelnetDevDeinit(VOID)
{
INT32 ret;
ret = system_console_deinit(TELNET);
if (ret != 0) {
PRINT_ERR("Telnet console deinit error.\n");
}
return ret;
}
#endif