457 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
		
		
			
		
	
	
			457 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
|  | /*
 | ||
|  |  * Copyright (c) 2006-2023, RT-Thread Development Team | ||
|  |  * | ||
|  |  * SPDX-License-Identifier: Apache-2.0 | ||
|  |  * | ||
|  |  * Change Logs: | ||
|  |  * Date           Author       Notes | ||
|  |  * 2023-11-13     Shell        init ver. | ||
|  |  */ | ||
|  | 
 | ||
|  | #define DBG_TAG "lwp.tty"
 | ||
|  | #define DBG_LVL DBG_INFO
 | ||
|  | #include <rtdbg.h>
 | ||
|  | 
 | ||
|  | #define TTY_CONF_INCLUDE_CCHARS
 | ||
|  | #include "tty_config.h"
 | ||
|  | #include "tty_internal.h"
 | ||
|  | #include "terminal.h"
 | ||
|  | 
 | ||
|  | /* configure option: timeout of tty drain wait */ | ||
|  | static int tty_drainwait = 5 * 60; | ||
|  | 
 | ||
|  | #define TTY_NAME_PREFIX "tty"
 | ||
|  | static char *alloc_device_name(const char *name) | ||
|  | { | ||
|  |     char *tty_dev_name; | ||
|  |     long name_buf_len = (sizeof(TTY_NAME_PREFIX) - 1) /* raw prefix */ | ||
|  |                         + rt_strlen(name)             /* custom name */ | ||
|  |                         + 1;                          /* tailing \0 */ | ||
|  | 
 | ||
|  |     tty_dev_name = rt_malloc(name_buf_len); | ||
|  |     if (tty_dev_name) | ||
|  |         sprintf(tty_dev_name, "%s%s", TTY_NAME_PREFIX, name); | ||
|  |     return tty_dev_name; | ||
|  | } | ||
|  | 
 | ||
|  | /* character device for tty */ | ||
|  | #ifdef RT_USING_DEVICE_OPS
 | ||
|  | const static struct rt_device_ops tty_dev_ops = { | ||
|  |     /* IO directly through device is not allowed */ | ||
|  | }; | ||
|  | #else
 | ||
|  | #error Must enable RT_USING_DEVICE_OPS in Kconfig
 | ||
|  | #endif
 | ||
|  | 
 | ||
|  | static int tty_fops_open(struct dfs_file *file) | ||
|  | { | ||
|  |     int rc; | ||
|  |     lwp_tty_t tp; | ||
|  |     rt_device_t device; | ||
|  |     int devtype = 0; /* unused */ | ||
|  | 
 | ||
|  |     if (file->vnode && file->vnode->data) | ||
|  |     { | ||
|  |         if (file->vnode->ref_count != 1) | ||
|  |         { | ||
|  |             rc = 0; | ||
|  |         } | ||
|  |         else | ||
|  |         { | ||
|  |             device = (rt_device_t)file->vnode->data; | ||
|  |             tp = rt_container_of(device, struct lwp_tty, parent); | ||
|  |             rc = bsd_ttydev_methods.d_open(tp, file->flags, devtype, | ||
|  |                                            rt_thread_self()); | ||
|  |         } | ||
|  |     } | ||
|  |     else | ||
|  |     { | ||
|  |         rc = -EINVAL; | ||
|  |     } | ||
|  | 
 | ||
|  |     return rc; | ||
|  | } | ||
|  | 
 | ||
|  | static int tty_fops_close(struct dfs_file *file) | ||
|  | { | ||
|  |     int rc; | ||
|  |     lwp_tty_t tp; | ||
|  |     rt_device_t device; | ||
|  |     int fflags = FFLAGS(file->flags); | ||
|  |     int devtype = 0; /* unused */ | ||
|  | 
 | ||
|  |     if (file->vnode && file->vnode->data) | ||
|  |     { | ||
|  |         if (file->vnode->ref_count != 1) | ||
|  |         { | ||
|  |             rc = 0; | ||
|  |         } | ||
|  |         else | ||
|  |         { | ||
|  |             device = (rt_device_t)file->vnode->data; | ||
|  |             tp = rt_container_of(device, struct lwp_tty, parent); | ||
|  |             rc = bsd_ttydev_methods.d_close(tp, fflags, devtype, rt_thread_self()); | ||
|  |         } | ||
|  |     } | ||
|  |     else | ||
|  |     { | ||
|  |         rc = -EINVAL; | ||
|  |     } | ||
|  | 
 | ||
|  |     return rc; | ||
|  | } | ||
|  | 
 | ||
|  | static int tty_fops_ioctl(struct dfs_file *file, int cmd, void *arg) | ||
|  | { | ||
|  |     int rc; | ||
|  |     lwp_tty_t tp; | ||
|  |     rt_device_t device; | ||
|  | 
 | ||
|  |     if (file->vnode && file->vnode->data) | ||
|  |     { | ||
|  |         device = (rt_device_t)file->vnode->data; | ||
|  |         tp = rt_container_of(device, struct lwp_tty, parent); | ||
|  |         rc = lwp_tty_ioctl_adapter(tp, cmd, file->flags, arg, rt_thread_self()); | ||
|  |     } | ||
|  |     else | ||
|  |     { | ||
|  |         rc = -EINVAL; | ||
|  |     } | ||
|  | 
 | ||
|  |     return rc; | ||
|  | } | ||
|  | 
 | ||
|  | static ssize_t tty_fops_read(struct dfs_file *file, void *buf, size_t count, | ||
|  |                              off_t *pos) | ||
|  | { | ||
|  |     ssize_t rc = 0; | ||
|  |     int error; | ||
|  |     struct uio uio; | ||
|  |     struct iovec iov; | ||
|  |     rt_device_t device; | ||
|  |     struct lwp_tty *tp; | ||
|  |     int ioflags; | ||
|  |     int oflags = file->flags; | ||
|  | 
 | ||
|  |     if (file->vnode && file->vnode->data) | ||
|  |     { | ||
|  |         device = (rt_device_t)file->vnode->data; | ||
|  |         tp = rt_container_of(device, struct lwp_tty, parent); | ||
|  | 
 | ||
|  |         /* setup ioflags */ | ||
|  |         ioflags = 0; | ||
|  |         if (oflags & O_NONBLOCK) | ||
|  |             ioflags |= IO_NDELAY; | ||
|  | 
 | ||
|  |         /* setup uio parameters */ | ||
|  |         iov.iov_base = (void *)buf; | ||
|  |         iov.iov_len = count; | ||
|  |         uio.uio_offset = file->fpos; | ||
|  |         uio.uio_resid = count; | ||
|  |         uio.uio_iov = &iov; | ||
|  |         uio.uio_iovcnt = 1; | ||
|  |         uio.uio_rw = UIO_READ; | ||
|  | 
 | ||
|  |         rc = count; | ||
|  |         error = bsd_ttydev_methods.d_read(tp, &uio, ioflags); | ||
|  |         rc -= uio.uio_resid; | ||
|  |         if (error) | ||
|  |         { | ||
|  |             LOG_D("%s: failed to write %d bytes of data. error code %d", | ||
|  |                   __func__, uio.uio_resid, error); | ||
|  |             rc = error; | ||
|  |         } | ||
|  | 
 | ||
|  |         /* reset file context */ | ||
|  |         file->fpos = uio.uio_offset; | ||
|  |     } | ||
|  | 
 | ||
|  |     if (rc) | ||
|  |         LOG_D("%s(len=%d, buf=%c \"%d\")", __func__, rc, *((char *)buf), | ||
|  |               *((char *)buf)); | ||
|  |     return rc; | ||
|  | } | ||
|  | 
 | ||
|  | static ssize_t tty_fops_write(struct dfs_file *file, const void *buf, | ||
|  |                               size_t count, off_t *pos) | ||
|  | { | ||
|  |     ssize_t rc = 0; | ||
|  |     int error; | ||
|  |     struct uio uio; | ||
|  |     struct iovec iov; | ||
|  |     rt_device_t device; | ||
|  |     struct lwp_tty *tp; | ||
|  |     int ioflags; | ||
|  |     int oflags = file->flags; | ||
|  | 
 | ||
|  |     if (file->vnode && file->vnode->data) | ||
|  |     { | ||
|  |         device = (rt_device_t)file->vnode->data; | ||
|  |         tp = rt_container_of(device, struct lwp_tty, parent); | ||
|  | 
 | ||
|  |         /* setup ioflags */ | ||
|  |         ioflags = 0; | ||
|  |         if (oflags & O_NONBLOCK) | ||
|  |             ioflags |= IO_NDELAY; | ||
|  | 
 | ||
|  |         /* setup uio parameters */ | ||
|  |         iov.iov_base = (void *)buf; | ||
|  |         iov.iov_len = count; | ||
|  |         uio.uio_offset = file->fpos; | ||
|  |         uio.uio_resid = count; | ||
|  |         uio.uio_iov = &iov; | ||
|  |         uio.uio_iovcnt = 1; | ||
|  |         uio.uio_rw = UIO_WRITE; | ||
|  | 
 | ||
|  |         rc = count; | ||
|  |         error = bsd_ttydev_methods.d_write(tp, &uio, ioflags); | ||
|  |         if (error) | ||
|  |         { | ||
|  |             rc = error; | ||
|  |             LOG_D("%s: failed to write %d bytes of data. error code %d", | ||
|  |                   __func__, uio.uio_resid, error); | ||
|  |         } | ||
|  |         else | ||
|  |         { | ||
|  |             rc -= uio.uio_resid; | ||
|  |         } | ||
|  | 
 | ||
|  |         /* reset file context */ | ||
|  |         file->fpos = uio.uio_offset; | ||
|  |     } | ||
|  |     return rc; | ||
|  | } | ||
|  | 
 | ||
|  | static int tty_fops_flush(struct dfs_file *file) | ||
|  | { | ||
|  |     return -EINVAL; | ||
|  | } | ||
|  | 
 | ||
|  | static off_t tty_fops_lseek(struct dfs_file *file, off_t offset, int wherece) | ||
|  | { | ||
|  |     return -EINVAL; | ||
|  | } | ||
|  | 
 | ||
|  | static int tty_fops_truncate(struct dfs_file *file, off_t offset) | ||
|  | { | ||
|  |     /**
 | ||
|  |      * regarding to POSIX.1, TRUNC is not supported for tty device. | ||
|  |      * return 0 always to make filesystem happy | ||
|  |      */ | ||
|  |     return 0; | ||
|  | } | ||
|  | 
 | ||
|  | static int tty_fops_poll(struct dfs_file *file, struct rt_pollreq *req) | ||
|  | { | ||
|  |     int rc; | ||
|  |     rt_device_t device; | ||
|  |     struct lwp_tty *tp; | ||
|  | 
 | ||
|  |     if (file->vnode && file->vnode->data) | ||
|  |     { | ||
|  |         device = (rt_device_t)file->vnode->data; | ||
|  |         tp = rt_container_of(device, struct lwp_tty, parent); | ||
|  | 
 | ||
|  |         rc = bsd_ttydev_methods.d_poll(tp, req, rt_thread_self()); | ||
|  |     } | ||
|  |     else | ||
|  |     { | ||
|  |         rc = -1; | ||
|  |     } | ||
|  | 
 | ||
|  |     return rc; | ||
|  | } | ||
|  | 
 | ||
|  | static int tty_fops_mmap(struct dfs_file *file, struct lwp_avl_struct *mmap) | ||
|  | { | ||
|  |     return -EINVAL; | ||
|  | } | ||
|  | 
 | ||
|  | static int tty_fops_lock(struct dfs_file *file, struct file_lock *flock) | ||
|  | { | ||
|  |     return -EINVAL; | ||
|  | } | ||
|  | 
 | ||
|  | static int tty_fops_flock(struct dfs_file *file, int operation, struct file_lock *flock) | ||
|  | { | ||
|  |     return -EINVAL; | ||
|  | } | ||
|  | 
 | ||
|  | static struct dfs_file_ops tty_file_ops = { | ||
|  |     .open = tty_fops_open, | ||
|  |     .close = tty_fops_close, | ||
|  |     .ioctl = tty_fops_ioctl, | ||
|  |     .read = tty_fops_read, | ||
|  |     .write = tty_fops_write, | ||
|  |     .flush = tty_fops_flush, | ||
|  |     .lseek = tty_fops_lseek, | ||
|  |     .truncate = tty_fops_truncate, | ||
|  |     .poll = tty_fops_poll, | ||
|  |     .mmap = tty_fops_mmap, | ||
|  |     .lock = tty_fops_lock, | ||
|  |     .flock = tty_fops_flock, | ||
|  | }; | ||
|  | 
 | ||
|  | rt_inline void device_setup(lwp_tty_t terminal) | ||
|  | { | ||
|  |     terminal->parent.type = RT_Device_Class_Char; | ||
|  | #ifdef RT_USING_DEVICE_OPS
 | ||
|  |     terminal->parent.ops = &tty_dev_ops; | ||
|  | #else
 | ||
|  | #error Must enable RT_USING_DEVICE_OPS in Kconfig
 | ||
|  | #endif
 | ||
|  | } | ||
|  | 
 | ||
|  | /* register TTY device */ | ||
|  | rt_err_t lwp_tty_register(lwp_tty_t terminal, const char *name) | ||
|  | { | ||
|  |     rt_err_t rc = -RT_ENOMEM; | ||
|  |     const char *tty_name; | ||
|  |     char *alloc_name; | ||
|  | 
 | ||
|  |     if (terminal->t_devsw->tsw_flags & TF_NOPREFIX) | ||
|  |     { | ||
|  |         alloc_name = RT_NULL; | ||
|  |         tty_name = name; | ||
|  |     } | ||
|  |     else | ||
|  |     { | ||
|  |         alloc_name = alloc_device_name(name); | ||
|  |         tty_name = alloc_name; | ||
|  |     } | ||
|  | 
 | ||
|  |     if (tty_name) | ||
|  |     { | ||
|  |         device_setup(terminal); | ||
|  |         rc = rt_device_register(&terminal->parent, tty_name, 0); | ||
|  |         if (rc == RT_EOK) | ||
|  |         { | ||
|  |             terminal->parent.fops = &tty_file_ops; | ||
|  | 
 | ||
|  |             LOG_D("%s() /dev/%s device registered", __func__, tty_name); | ||
|  |         } | ||
|  | 
 | ||
|  |         rt_free(alloc_name); | ||
|  |     } | ||
|  |     return rc; | ||
|  | } | ||
|  | 
 | ||
|  | static void tty_init_termios(lwp_tty_t tp) | ||
|  | { | ||
|  |     struct termios *t = &tp->t_termios_init_in; | ||
|  | 
 | ||
|  |     t->c_cflag = TTYDEF_CFLAG; | ||
|  |     t->c_iflag = TTYDEF_IFLAG; | ||
|  |     t->c_lflag = TTYDEF_LFLAG; | ||
|  |     t->c_oflag = TTYDEF_OFLAG; | ||
|  |     t->__c_ispeed = TTYDEF_SPEED; | ||
|  |     t->__c_ospeed = TTYDEF_SPEED; | ||
|  | 
 | ||
|  |     memcpy(&t->c_cc, tty_ctrl_charset, | ||
|  |            sizeof(tty_ctrl_charset) / sizeof(tty_ctrl_charset[0])); | ||
|  | 
 | ||
|  | #ifdef USING_BSD_INIT_LOCK_DEVICE
 | ||
|  |     tp->t_termios_init_out = *t; | ||
|  | #endif /* USING_BSD_INIT_LOCK_DEVICE */
 | ||
|  | } | ||
|  | 
 | ||
|  | lwp_tty_t lwp_tty_create_ext(lwp_ttydevsw_t handle, void *softc, | ||
|  |                              rt_mutex_t custom_mtx) | ||
|  | { | ||
|  |     lwp_tty_t tp; | ||
|  | 
 | ||
|  |     tp = rt_calloc(1, sizeof(struct lwp_tty) | ||
|  |     #ifdef USING_BSD_SIGINFO
 | ||
|  |             + LWP_TTY_PRBUF_SIZE | ||
|  |     #endif
 | ||
|  |             ); | ||
|  | 
 | ||
|  |     if (!tp) | ||
|  |         return tp; | ||
|  | 
 | ||
|  |     bsd_devsw_init(handle); | ||
|  | 
 | ||
|  | #ifdef USING_BSD_SIGINFO
 | ||
|  |     tp->t_prbufsz = LWP_TTY_PRBUF_SIZE; | ||
|  | #endif
 | ||
|  |     tp->t_devsw = handle; | ||
|  |     tp->t_devswsoftc = softc; | ||
|  |     tp->t_flags = handle->tsw_flags; | ||
|  |     tp->t_drainwait = tty_drainwait; | ||
|  | 
 | ||
|  |     tty_init_termios(tp); | ||
|  | 
 | ||
|  |     cv_init(&tp->t_inwait, "ttyin"); | ||
|  |     cv_init(&tp->t_outwait, "ttyout"); | ||
|  |     cv_init(&tp->t_outserwait, "ttyosr"); | ||
|  |     cv_init(&tp->t_bgwait, "ttybg"); | ||
|  |     cv_init(&tp->t_dcdwait, "ttydcd"); | ||
|  | 
 | ||
|  |     rt_wqueue_init(&tp->t_inpoll); | ||
|  |     rt_wqueue_init(&tp->t_outpoll); | ||
|  | 
 | ||
|  |     /* Allow drivers to use a custom mutex to lock the TTY. */ | ||
|  |     if (custom_mtx != NULL) | ||
|  |     { | ||
|  |         tp->t_mtx = custom_mtx; | ||
|  |     } | ||
|  |     else | ||
|  |     { | ||
|  |         tp->t_mtx = &tp->t_mtxobj; | ||
|  |         rt_mutex_init(&tp->t_mtxobj, "ttydev", RT_IPC_FLAG_PRIO); | ||
|  |     } | ||
|  | 
 | ||
|  | #ifdef USING_BSD_POLL
 | ||
|  |     knlist_init_mtx(&tp->t_inpoll.si_note, tp->t_mtx); | ||
|  |     knlist_init_mtx(&tp->t_outpoll.si_note, tp->t_mtx); | ||
|  | #endif
 | ||
|  | 
 | ||
|  |     return tp; | ||
|  | } | ||
|  | 
 | ||
|  | lwp_tty_t lwp_tty_create(lwp_ttydevsw_t handle, void *softc) | ||
|  | { | ||
|  |     return lwp_tty_create_ext(handle, softc, NULL); | ||
|  | } | ||
|  | 
 | ||
|  | void lwp_tty_delete(lwp_tty_t tp) | ||
|  | { | ||
|  |     /*
 | ||
|  |      * ttyydev_leave() usually frees the i/o queues earlier, but it is | ||
|  |      * not always called between queue allocation and here.  The queues | ||
|  |      * may be allocated by ioctls on a pty control device without the | ||
|  |      * corresponding pty slave device ever being open, or after it is | ||
|  |      * closed. | ||
|  |      */ | ||
|  |     ttyinq_free(&tp->t_inq); | ||
|  |     ttyoutq_free(&tp->t_outq); | ||
|  |     rt_wqueue_wakeup_all(&tp->t_inpoll, (void *)POLLHUP); | ||
|  |     rt_wqueue_wakeup_all(&tp->t_outpoll, (void *)POLLHUP); | ||
|  | 
 | ||
|  | #ifdef USING_BSD_POLL
 | ||
|  |     knlist_destroy(&tp->t_inpoll.si_note); | ||
|  |     knlist_destroy(&tp->t_outpoll.si_note); | ||
|  | #endif
 | ||
|  | 
 | ||
|  |     cv_destroy(&tp->t_inwait); | ||
|  |     cv_destroy(&tp->t_outwait); | ||
|  |     cv_destroy(&tp->t_bgwait); | ||
|  |     cv_destroy(&tp->t_dcdwait); | ||
|  |     cv_destroy(&tp->t_outserwait); | ||
|  | 
 | ||
|  |     if (tp->t_mtx == &tp->t_mtxobj) | ||
|  |         rt_mutex_detach(&tp->t_mtxobj); | ||
|  |     ttydevsw_free(tp); | ||
|  |     rt_device_unregister(&tp->parent); | ||
|  |     rt_free(tp); | ||
|  | } | ||
|  | 
 | ||
|  | /*
 | ||
|  |  * Report on state of foreground process group. | ||
|  |  */ | ||
|  | void tty_info(struct lwp_tty *tp) | ||
|  | { | ||
|  |     /* TODO */ | ||
|  |     return; | ||
|  | } |