最新资讯

  • 理解UART 子系统:Linux Kernel 4.9.88 中的核心结构体与设计详解

理解UART 子系统:Linux Kernel 4.9.88 中的核心结构体与设计详解

2025-04-29 09:00:12 0 阅读

往期内容

本专栏往期内容:Uart子系统

  1. UART串口硬件介绍
  2. 深入理解TTY体系:设备节点与驱动程序框架详解
  3. Linux串口应用编程:从UART到GPS模块及字符设备驱动

interrupt子系统专栏:

  1. 专栏地址:interrupt子系统
  2. Linux 链式与层级中断控制器讲解:原理与驱动开发
    – 末片,有专栏内容观看顺序

pinctrl和gpio子系统专栏:

  1. 专栏地址:pinctrl和gpio子系统

  2. 编写虚拟的GPIO控制器的驱动程序:和pinctrl的交互使用

    – 末片,有专栏内容观看顺序

input子系统专栏:

  1. 专栏地址:input子系统
  2. input角度:I2C触摸屏驱动分析和编写一个简单的I2C驱动程序
    – 末片,有专栏内容观看顺序

I2C子系统专栏:

  1. 专栏地址:IIC子系统
  2. 具体芯片的IIC控制器驱动程序分析:i2c-imx.c-CSDN博客
    – 末篇,有专栏内容观看顺序

总线和设备树专栏:

  1. 专栏地址:总线和设备树
  2. 设备树与 Linux 内核设备驱动模型的整合-CSDN博客
    – 末篇,有专栏内容观看顺序


目录

  • 往期内容
  • 1.框图
  • 2.uart_port
  • 3.uart_driver
  • 4.tty_port
  • 5.tty_driver
  • 6.uart_state
  • 7.tty_operations
  • 8.uart_ops
  • 9.联系
  • 10.tty_truct
  • 11.tty_port_operaion

1.框图

2.uart_port

uart_port 结构体包含 UART 端口的具体信息,包括寄存器地址、锁、端口设置等,UART 核心和底层驱动共同使用此结构体来实现端口的配置、控制和数据传输。下面的成员是有所省略的。

Linux-4.9.88includelinuxserial_core.h
/**
 * struct uart_port - 描述 UART 端口的硬件配置及控制方法
 * @lock: 自旋锁,用于保护端口数据访问的同步
 * @iobase: 端口的 IO 基地址,用于 IO 映射
 * @membase: 端口的内存映射基地址,用于内存映射方式
 * @serial_in: 函数指针,用于读取寄存器中的数据
 * @serial_out: 函数指针,用于写入数据到寄存器
 * @set_termios: 配置串口的波特率、数据位等参数
 * @get_mctrl: 获取控制信号状态(如 CTS、DCD 等)
 * @set_mctrl: 设置控制信号(如 RTS、DTR 等)
 * @startup: 启动 UART 端口
 * @shutdown: 关闭 UART 端口
 * @throttle: 降速处理,当缓冲区快满时限制数据传输
 * @unthrottle: 恢复数据传输
 * @handle_irq: 中断处理函数,处理端口的中断请求
 * @pm: 电源管理回调,用于切换电源状态
 * @handle_break: 处理中断信号
 * @irq: 端口的 IRQ 编号
 * @irqflags: IRQ 标志,用于设置中断类型
 * @uartclk: UART 基准时钟频率,用于计算波特率
 * @fifosize: 发送 FIFO 的大小
 * @flags: 端口的控制标志,包含流控制、低延迟等设置
 * @status: UART 端口的状态标志
 * @ops: 指向 UART 操作的结构体指针,包含各种操作函数
 * @dev: 指向此 UART 端口的设备结构体指针
 * @private_data: 通用平台数据指针,用于存储私有数据
 */
struct uart_port {
    spinlock_t lock;                  // 端口锁,用于保护并发访问
    unsigned long iobase;             // IO 基地址
    unsigned char __iomem *membase;   // 内存映射的基地址
    unsigned int (*serial_in)(struct uart_port *, int);  // 读取寄存器
    void (*serial_out)(struct uart_port *, int, int);    // 写入寄存器
    void (*set_termios)(struct uart_port *, struct ktermios *new, struct ktermios *old); // 配置串口
    unsigned int (*get_mctrl)(struct uart_port *);       // 获取控制信号
    void (*set_mctrl)(struct uart_port *, unsigned int); // 设置控制信号
    int (*startup)(struct uart_port *port);              // 启动 UART 端口
    void (*shutdown)(struct uart_port *port);            // 关闭 UART 端口
    void (*throttle)(struct uart_port *port);            // 数据传输限制
    void (*unthrottle)(struct uart_port *port);          // 恢复数据传输
    int (*handle_irq)(struct uart_port *);               // 中断处理函数
    void (*pm)(struct uart_port *, unsigned int state, unsigned int old); // 电源管理
    void (*handle_break)(struct uart_port *);            // 处理中断信号
    int (*rs485_config)(struct uart_port *, struct serial_rs485 *rs485); // RS485 配置
    unsigned int irq;                     // 中断请求编号
    unsigned long irqflags;               // 中断标志
    unsigned int uartclk;                 // 基准时钟频率
    unsigned int fifosize;                // FIFO 缓冲区大小
    unsigned char x_char;                 // XON/XOFF 字符
    unsigned char regshift;               // 寄存器偏移量
    unsigned char iotype;                 // IO 访问类型
    unsigned int read_status_mask;        // 读取状态掩码
    unsigned int ignore_status_mask;      // 忽略状态掩码
    struct uart_state *state;             // 指向 UART 状态
    struct uart_icount icount;            // UART 统计数据
    struct console *cons;                 // 控制台指针
    upf_t flags;                          // UART 端口的标志
    upstat_t status;                      // 端口状态标志
    int hw_stopped;                       // 硬件停止状态
    unsigned int mctrl;                   // 调制解调控制状态
    unsigned int timeout;                 // 超时设置
    unsigned int type;                    // UART 端口类型
    const struct uart_ops *ops;           // UART 操作接口
    unsigned int custom_divisor;          // 自定义波特率分频器
    unsigned int line;                    // UART 端口编号
    unsigned int minor;                   // 次设备号
    resource_size_t mapbase;              // IO 内存基地址
    resource_size_t mapsize;              // IO 内存大小
    struct device *dev;                   // 设备指针
    unsigned char hub6;                   // Hub6 ISA 卡的配置
    unsigned char suspended;              // 挂起状态
    unsigned char irq_wake;               // IRQ 唤醒状态
    struct attribute_group *attr_group;   // 端口的特定属性
    const struct attribute_group **tty_groups; // 全部 tty 属性
    struct serial_rs485 rs485;            // RS485 协议配置
    void *private_data;                   // 平台特定的私有数据指针
};

3.uart_driver

uart_driver 结构体包含了 UART 驱动程序的相关信息,例如驱动名称、主次设备号等,用于注册和管理 UART 驱动。

Linux-4.9.88includelinuxserial_core.h
/**
 * struct uart_driver - 描述 UART 驱动信息
 * @owner: 指向驱动程序模块的指针,通常设置为 THIS_MODULE
 * @driver_name: 驱动程序的名称,通常用于 sysfs 中
 * @dev_name: 设备名称前缀,用于生成设备节点名称(如 ttyS)
 * @major: 主设备号,用于指定 UART 驱动的主设备号
 * @minor: 次设备号,通常用于为多个 UART 设备提供编号
 * @nr: 最大支持的 UART 端口数量
 * @cons: 与驱动相关联的控制台设备
 * @state: 指向每个 UART 端口的状态信息,通常由 UART 核心初始化
 * @tty_driver: 指向 tty_driver 结构体的指针,表示与此 UART 驱动关联的 TTY 驱动
 */
struct uart_driver {
    struct module *owner;              // 驱动程序模块的指针
    const char *driver_name;           // UART 驱动的名称
    const char *dev_name;              // 设备名称前缀
    int major;                         // 主设备号
    int minor;                         // 次设备号
    int nr;                            // 支持的最大 UART 端口数
    struct console *cons;              // 控制台设备结构体指针

    // 以下字段应由 UART 核心初始化,底层驱动不应直接操作
    struct uart_state *state;          // UART 状态信息
    struct tty_driver *tty_driver;     // TTY 驱动的指针
};

4.tty_port

tty_port 结构体管理 TTY 端口的资源和状态,包括等待队列、互斥锁、缓冲区等,用于简化驱动层对 TTY 端口的管理。

Linux-4.9.88includelinux	ty.h
/**
 * struct tty_port - 描述 TTY 端口的信息和状态
 * @buf: 缓冲区头,用于管理端口的数据缓冲区,内部锁定保护
 * @tty: 指向 tty_struct 的指针,用于指向已分配的 TTY 结构体(回指)
 * @itty: 内部指向 tty_struct 的指针,用于指向内部TTY 结构体(内部回指)
 * @ops: 指向 tty_port_operations 结构体的指针,定义了端口的操作函数
 * @lock: 自旋锁,用于保护 tty 字段的访问
 * @blocked_open: 阻塞打开标志,用于表示端口是否正在等待打开
 * @count: 使用计数,表示端口被打开的次数
 * @open_wait: 等待队列,用于阻塞打开的进程
 * @delta_msr_wait: 调制解调器状态变化等待队列,用于等待调制解调器状态改变
 * @flags: 用户 TTY 标志,表示端口的属性,例如异步标志
 * @iflags: 内部标志,用于表示端口的内部状态(TTY_PORT_ 开头)
 * @console: 标志位,表示该端口是否是控制台
 * @low_latency: 可选标志位,表示是否调优以降低延迟
 * @mutex: 互斥锁,用于端口锁定
 * @buf_mutex: 缓冲区分配锁,用于管理缓冲区的分配
 * @xmit_buf: 可选发送缓冲区,用于保存传输的数据
 * @close_delay: 关闭端口时的延迟,单位为毫秒
 * @closing_wait: 输出完成的延迟时间
 * @drain_delay: 排空延迟,当无需基于时间的纯排空时设置为零,否则设置为 FIFO 大小
 * @kref: 引用计数器,用于控制 tty_port 结构体的引用计数
 */
struct tty_port {
    struct tty_bufhead buf;           // 数据缓冲区头,管理端口的缓冲区
    struct tty_struct *tty;           // 指向 tty_struct 的指针(回指)
    struct tty_struct *itty;          // 内部指向 tty_struct 的指针(内部回指)
    const struct tty_port_operations *ops; // 端口的操作函数指针
    spinlock_t lock;                  // 自旋锁,保护 tty 字段
    int blocked_open;                 // 阻塞打开标志
    int count;                        // 使用计数
    wait_queue_head_t open_wait;      // 打开等待队列
    wait_queue_head_t delta_msr_wait; // 调制解调器状态变化等待队列
    unsigned long flags;              // 用户 TTY 标志(如 ASYNC_)
    unsigned long iflags;             // 内部标志(如 TTY_PORT_)
    unsigned char console:1;          // 是否是控制台
    unsigned char low_latency:1;      // 是否调优为低延迟
    struct mutex mutex;               // 端口互斥锁
    struct mutex buf_mutex;           // 缓冲区分配锁
    unsigned char *xmit_buf;          // 可选的发送缓冲区
    unsigned int close_delay;         // 关闭端口时的延迟(毫秒)
    unsigned int closing_wait;        // 输出完成延迟时间
    int drain_delay;                  // 排空延迟
    struct kref kref;                 // 引用计数器
};

5.tty_driver

tty_driver 结构体负责管理 TTY 驱动的基本信息,包括驱动名称、设备编号、TTY 数量等。同时,定义了特定的驱动操作函数接口,以便 TTY 子系统调用这些函数来操作设备。

/**
 * struct tty_driver - 描述 TTY 驱动的信息及操作函数
 * @magic: 魔数,用于识别 TTY 驱动结构体的合法性
 * @kref: 引用计数管理,用于控制结构体的生命周期
 * @cdevs: 指向字符设备的指针数组
 * @owner: 指向驱动模块的指针,通常为 THIS_MODULE
 * @driver_name: 驱动程序的名称,用于标识驱动
 * @name: TTY 设备名称,用于生成设备节点名称(例如 ttyS)
 * @name_base: 名称基址,用于计算 TTY 设备的完整名称
 * @major: 主设备号,用于标识驱动
 * @minor_start: 起始次设备号,通常用于为多个 TTY 设备提供编号
 * @num: 分配的设备数量
 * @type: 驱动类型,用于区分不同类型的 TTY 驱动
 * @subtype: 驱动子类型,进一步区分 TTY 驱动的类型
 * @init_termios: 初始终端设置,用于初始化 TTY 的终端属性
 * @flags: 驱动标志,用于标识驱动的特殊属性
 * @proc_entry: 指向 proc 文件系统的条目指针
 * @other: 仅用于 PTY 驱动,指向配对的 TTY 驱动
 * @ttys: 指向 tty_struct 指针数组,用于管理所有 TTY 设备
 * @ports: 指向 tty_port 指针数组,指向所有端口信息
 * @termios: 指向 ktermios 指针数组,保存每个 TTY 的终端属性
 * @driver_state: 驱动状态指针,用于存储特定的驱动状态信息
 * @ops: 指向 tty_operations 的指针,包含驱动操作函数
 * @tty_drivers: TTY 驱动列表,用于连接多个 TTY 驱动
 */
struct tty_driver {
    int magic;                      // 魔数,确保结构体合法性
    struct kref kref;               // 引用计数管理
    struct cdev **cdevs;            // 字符设备指针数组
    struct module *owner;           // 指向驱动模块(通常为 THIS_MODULE)
    const char *driver_name;        // 驱动程序的名称
    const char *name;               // TTY 设备名称
    int name_base;                  // 名称基址
    int major;                      // 主设备号
    int minor_start;                // 起始次设备号
    unsigned int num;               // 分配的设备数量
    short type;                     // 驱动类型
    short subtype;                  // 驱动子类型
    struct ktermios init_termios;   // 初始终端设置
    unsigned long flags;            // 驱动标志
    struct proc_dir_entry *proc_entry; // /proc 文件系统条目
    struct tty_driver *other;       // 配对的 TTY 驱动(仅用于 PTY 驱动)

    // 指向 TTY 数据结构的指针
    struct tty_struct **ttys;       // TTY 结构体指针数组
    struct tty_port **ports;        // TTY 端口指针数组
    struct ktermios **termios;      // TTY 的终端设置
    void *driver_state;             // 驱动状态信息

    // 驱动方法
    const struct tty_operations *ops; // 驱动操作函数
    struct list_head tty_drivers;   // TTY 驱动列表
};

6.uart_state

uart_driver结构体的成员

Linux-4.9.88includelinuxserial_core.h
/*
 * This is the state information which is persistent across opens.
 */
struct uart_state {
	struct tty_port		port;        // 表示与TTY设备相关的端口信息,包含串口的输入输出控制功能。

	enum uart_pm_state	pm_state;   // UART的电源管理状态,指示串口的当前电源状态(如:活动、休眠等)。

	struct circ_buf		xmit;       // 用于存储要发送的字符的循环缓冲区,支持异步传输。

	atomic_t		refcount;    // 引用计数,用于跟踪当前有多少个打开的句柄指向此状态结构体。

	wait_queue_head_t	remove_wait; // 用于进程等待的队列头,如果设备正在被移除,则相应的进程可以等待。

	struct uart_port	*uart_port; // 指向具体的UART端口结构体,包含硬件相关信息,如基地址、IRQ等。
};


	truct circ_buf		xmit;       // 用于存储要发送的字符的循环缓冲区,支持异步传输。

	atomic_t		refcount;    // 引用计数,用于跟踪当前有多少个打开的句柄指向此状态结构体。

	wait_queue_head_t	remove_wait; // 用于进程等待的队列头,如果设备正在被移除,则相应的进程可以等待。

	struct uart_port	*uart_port; // 指向具体的UART端口结构体,包含硬件相关信息,如基地址、IRQ等。
};

uart_state结构体用于表示UART(通用异步收发传输)设备的状态信息,保存了关于设备操作的多个重要参数,如当前的电源管理状态、待发送的数据缓冲区、引用计数等。这些信息在设备打开和关闭时保持不变。

7.tty_operations

tty_driver结构体的成员

struct tty_operations {
	struct tty_struct * (*lookup)(struct tty_driver *driver, // 查找TTY结构体的操作函数
			struct file *filp, int idx);             // 根据驱动、文件和索引返回相应的tty_struct。

	int  (*install)(struct tty_driver *driver,              // 安装TTY设备的操作函数
			struct tty_struct *tty);                   // 初始化tty_struct。

	void (*remove)(struct tty_driver *driver,               // 移除TTY设备的操作函数
			struct tty_struct *tty);                   // 清理tty_struct资源。

	int  (*open)(struct tty_struct * tty,                   // 打开TTY设备的操作函数
			struct file * filp);                        // 对应于打开文件的系统调用。

	void (*close)(struct tty_struct * tty,                  // 关闭TTY设备的操作函数
			struct file * filp);                        // 对应于关闭文件的系统调用。

	void (*shutdown)(struct tty_struct *tty);               // 关闭TTY设备并清理资源的操作函数。

	void (*cleanup)(struct tty_struct *tty);                // 清理TTY设备资源的操作函数。

	int  (*write)(struct tty_struct * tty,                  // 写入数据到TTY设备的操作函数
		      const unsigned char *buf, int count);     // 将缓冲区中的数据写入tty。

	int  (*put_char)(struct tty_struct *tty,               // 向TTY设备发送单个字符的操作函数
		      unsigned char ch);                         // 字符。

	void (*flush_chars)(struct tty_struct *tty);           // 刷新TTY设备中的字符(清空缓冲区)。

	int  (*write_room)(struct tty_struct *tty);            // 获取TTY设备的写入空间大小。

	int  (*chars_in_buffer)(struct tty_struct *tty);       // 获取TTY设备缓冲区中字符的数量。

	int  (*ioctl)(struct tty_struct *tty,                   // 处理TTY设备的控制命令的操作函数
		    unsigned int cmd, unsigned long arg);      // 控制命令和参数。

	long (*compat_ioctl)(struct tty_struct *tty,           // 处理兼容性的ioctl命令的操作函数
			     unsigned int cmd, unsigned long arg);

	void (*set_termios)(struct tty_struct *tty,            // 设置TTY设备的终端属性
			     struct ktermios * old);             // 旧的终端属性。

	void (*throttle)(struct tty_struct * tty);             // 暂停TTY设备的输出。

	void (*unthrottle)(struct tty_struct * tty);           // 恢复TTY设备的输出。

	void (*stop)(struct tty_struct *tty);                  // 停止TTY设备的操作。

	void (*start)(struct tty_struct *tty);                 // 启动TTY设备的操作。

	void (*hangup)(struct tty_struct *tty);                // 挂断TTY设备的操作。

	int (*break_ctl)(struct tty_struct *tty, int state);   // 控制TTY设备的断开状态。

	void (*flush_buffer)(struct tty_struct *tty);          // 刷新TTY设备的缓冲区。

	void (*set_ldisc)(struct tty_struct *tty);             // 设置TTY设备的线路设备。

	void (*wait_until_sent)(struct tty_struct *tty, int timeout); // 等待TTY设备发送完毕。

	void (*send_xchar)(struct tty_struct *tty, char ch);   // 向TTY设备发送控制字符。

	int (*tiocmget)(struct tty_struct *tty);                // 获取TTY设备的状态位。

	int (*tiocmset)(struct tty_struct *tty,                 // 设置TTY设备的状态位。
			unsigned int set, unsigned int clear);    // 要设置和清除的位。

	int (*resize)(struct tty_struct *tty,                  // 调整TTY设备窗口大小的操作函数。
			struct winsize *ws);                       // 新的窗口大小。

	int (*set_termiox)(struct tty_struct *tty,             // 设置扩展终端参数的操作函数。
			struct termiox *tnew);                    // 新的终端参数。

	int (*get_icount)(struct tty_struct *tty,              // 获取TTY设备中输入计数的操作函数。
				struct serial_icounter_struct *icount); // 输入计数结构体。

#ifdef CONFIG_CONSOLE_POLL
	int (*poll_init)(struct tty_driver *driver, int line, char *options); // 初始化轮询的操作函数。
	int (*poll_get_char)(struct tty_driver *driver, int line);             // 获取字符的轮询操作函数。
	void (*poll_put_char)(struct tty_driver *driver, int line, char ch);   // 发送字符的轮询操作函数。
#endif

	const struct file_operations *proc_fops;               // 指向文件操作的结构体,用于处理proc文件系统。
};

tty_operations结构体定义了与TTY(终端)设备相关的操作函数的指针。这些函数用于管理TTY设备的生命周期、数据读写、状态控制等。每个函数的实现可以根据不同的TTY驱动进行定制,提供与硬件相关的操作接口。

8.uart_ops

uart_port结构体的成员

Linux-4.9.88includelinuxserial_core.h

/*
 * This structure describes all the operations that can be done on the
 * physical hardware.  See Documentation/serial/driver for details.
 */
struct uart_ops {
	unsigned int	(*tx_empty)(struct uart_port *); // 检查传输寄存器是否为空的操作函数。

	void		(*set_mctrl)(struct uart_port *, unsigned int mctrl); // 设置控制线的操作函数。

	unsigned int	(*get_mctrl)(struct uart_port *); // 获取控制线状态的操作函数。

	void		(*stop_tx)(struct uart_port *); // 停止数据发送的操作函数。

	void		(*start_tx)(struct uart_port *); // 开始数据发送的操作函数。

	void		(*throttle)(struct uart_port *); // 暂停数据发送的操作函数。

	void		(*unthrottle)(struct uart_port *); // 恢复数据发送的操作函数。

	void		(*send_xchar)(struct uart_port *, char ch); // 发送一个特定字符的操作函数。

	void		(*stop_rx)(struct uart_port *); // 停止数据接收的操作函数。

	void		(*enable_ms)(struct uart_port *); // 启用调制解调器信号的操作函数。

	void		(*break_ctl)(struct uart_port *, int ctl); // 控制断开信号的操作函数。

	int		(*startup)(struct uart_port *); // 启动UART设备的操作函数。

	void		(*shutdown)(struct uart_port *); // 关闭UART设备的操作函数。

	void		(*flush_buffer)(struct uart_port *); // 刷新缓冲区的操作函数。

	void		(*set_termios)(struct uart_port *, struct ktermios *new,
				       struct ktermios *old); // 设置终端属性的操作函数。

	void		(*set_ldisc)(struct uart_port *, struct ktermios *); // 设置线路设备的操作函数。

	void		(*pm)(struct uart_port *, unsigned int state,
			      unsigned int oldstate); // 进行电源管理的操作函数。

	/*
	 * Return a string describing the type of the port
	 */
	const char	*(*type)(struct uart_port *); // 返回描述端口类型的字符串的操作函数。

	/*
	 * Release IO and memory resources used by the port.
	 * This includes iounmap if necessary.
	 */
	void		(*release_port)(struct uart_port *); // 释放端口使用的IO和内存资源的操作函数。

	/*
	 * Request IO and memory resources used by the port.
	 * This includes iomapping the port if necessary.
	 */
	int		(*request_port)(struct uart_port *); // 请求端口使用的IO和内存资源的操作函数。

	void		(*config_port)(struct uart_port *, int); // 配置端口的操作函数。

	int		(*verify_port)(struct uart_port *, struct serial_struct *); // 验证端口配置的操作函数。

	int		(*ioctl)(struct uart_port *, unsigned int, unsigned long); // 处理IO控制命令的操作函数。

#ifdef CONFIG_CONSOLE_POLL
	int		(*poll_init)(struct uart_port *); // 初始化轮询的操作函数。
	void		(*poll_put_char)(struct uart_port *, unsigned char); // 轮询发送字符的操作函数。
	int		(*poll_get_char)(struct uart_port *); // 轮询接收字符的操作函数。
#endif
};
  • uart_ops 结构体用于定义与物理UART硬件相关的所有操作。这些操作涵盖了UART设备的基本功能,如数据传输、控制线管理、端口配置、资源请求等。
  • 每个函数指针代表一种操作,具体的实现由对应的UART驱动提供。这样,uart_ops 结构体使得不同的UART硬件能够通过统一的接口进行操作,从而简化了驱动程序的开发。
  • 例如,start_txstop_tx 用于控制数据发送,而 set_termios 则用于配置UART的终端参数。这些函数的实现可以根据具体的硬件特性进行调整,确保在不同平台上能有效地控制UART设备。

需要额外注意的是不要和tty_operations混淆,因为驱动程序中经常这样去定义:struct tty_operations uar_opsstruct uart_ops imx_xxxxx,是不是很容易混淆,这点一定要注意。

9.联系

10.tty_truct

/*
 * 该结构体存储与TTY相关的所有状态信息,当TTY处于打开状态时。
 * 由于termios状态在TTY关闭后仍应保留(例如波特率等),
 * 因此不会在此存储,而是存储指向真实状态的指针。
 * winsize结构可能应采取相同的处理,但(1)默认的80x24通常是正确的,
 * 并且(2)它最常由窗口系统使用,每次创建或调整窗口大小时都会设置正确的大小。
 * 						- TYT, 9/14/92
 */

struct tty_operations; // 前向声明TTY操作结构

struct tty_struct {
	int magic; // 结构的魔术值,用于识别TTY结构的有效性
	struct kref kref; // 引用计数,用于管理TTY结构的生命周期
	struct device *dev; // 关联的设备结构指针
	struct tty_driver *driver; // 指向驱动程序的指针
	const struct tty_operations *ops; // TTY操作的指针

	int index; // TTY设备的索引

	/* 保护线路纪律变化:锁定TTY而不是PTY */
	struct ld_semaphore ldisc_sem; // 用于保护线路纪律的信号量
	struct tty_ldisc *ldisc; // 指向线路纪律的指针

	struct mutex atomic_write_lock; // 保护原子写入的互斥锁
	struct mutex legacy_mutex; // 旧代码的互斥锁
	struct mutex throttle_mutex; // 用于控制流的互斥锁
	struct rw_semaphore termios_rwsem; // 保护termios的读写信号量
	struct mutex winsize_mutex; // 保护窗口大小的互斥锁
	spinlock_t ctrl_lock; // 控制状态的自旋锁
	spinlock_t flow_lock; // 控制流的自旋锁

	/* termios值由termios_rwsem保护 */
	struct ktermios termios, termios_locked; // termios结构,保存串口设置
	struct termiox *termiox; // 可能为NULL,表示不支持的扩展设置

	char name[64]; // TTY的名称
	struct pid *pgrp; // 进程组ID,由ctrl_lock保护
	struct pid *session; // 会话ID
	unsigned long flags; // 状态标志位
	int count; // 当前打开的引用计数

	struct winsize winsize; // 窗口大小,受winsize_mutex保护
	unsigned long stopped:1, // 表示流是否停止
		      flow_stopped:1, // 表示流是否已停止
		      unused:BITS_PER_LONG - 2; // 未使用的位

	int hw_stopped; // 硬件停止标志

	unsigned long ctrl_status:8, // 控制状态
		      packet:1, // 表示是否为数据包
		      unused_ctrl:BITS_PER_LONG - 9; // 未使用的位

	unsigned int receive_room; // 接收队列的可用字节数
	int flow_change; // 流控制变化标志

	struct tty_struct *link; // 指向其他TTY结构的链接
	struct fasync_struct *fasync; // 异步通知结构
	int alt_speed; // 用于38400 bps的魔法替代

	wait_queue_head_t write_wait; // 写等待队列头
	wait_queue_head_t read_wait; // 读等待队列头

	struct work_struct hangup_work; // 处理挂断的工作结构
	void *disc_data; // 线路纪律的私有数据
	void *driver_data; // 驱动程序的私有数据

	spinlock_t files_lock; // 保护tty_files列表的自旋锁
	struct list_head tty_files; // 关联的打开文件列表

#define N_TTY_BUF_SIZE 4096 // 定义TTY缓冲区大小

	int closing; // 指示TTY是否正在关闭
	unsigned char *write_buf; // 写入缓冲区指针
	int write_cnt; // 写入计数

	/* 如果TTY有待处理的do_SAK,将其排队在此 - akpm */
	struct work_struct SAK_work; // 处理SAK的工作结构
	struct tty_port *port; // 指向TTY端口的指针
};
  • 此结构体用于保存与TTY设备的状态信息和相关数据。它在TTY设备打开时保持活动状态。
  • 包含对TTY设备的各种信息,如引用计数、设备驱动、操作函数、窗口大小、流控制状态等。
  • 通过不同的锁和信号量保护其各个部分的并发访问,确保在多线程环境中操作的安全性。

11.tty_port_operaion

tty_port的成员

struct tty_port_operations {
	/* 
	 * 返回 1 如果载波(carrier)信号被抬起。
	 * 这个函数用于检查串行端口的载波状态。
	 */
	int (*carrier_raised)(struct tty_port *port);

	/* 
	 * 控制 DTR(数据终端就绪)和 RTS(请求发送)线路。
	 * 通过设置参数来抬起或放下这两条控制线路。
	 */
	void (*dtr_rts)(struct tty_port *port, int raise);

	/* 
	 * 当最后一个关闭操作完成或挂起操作结束时调用,
	 * 仅在端口已初始化的情况下调用。
	 * 注意:不要在此处释放资源,因为它是在端口互斥锁下调用的,
	 * 用于序列化激活和关闭操作。
	 */
	void (*shutdown)(struct tty_port *port);

	/* 
	 * 在 tty_port_open 函数中在端口互斥锁下调用,
	 * 用于初始化设备。长远来看,希望将 tty 参数移出此函数,
	 * 以提高通用性(特别是对于控制台)。
	 */
	int (*activate)(struct tty_port *port, struct tty_struct *tty);

	/* 
	 * 在端口的最后一次释放(put)时调用。
	 * 这个函数用于清理与端口相关的资源。
	 */
	void (*destruct)(struct tty_port *port);
};

tty_port_operations 是一个函数指针结构体,定义了一组与TTY端口操作相关的回调函数。这些函数由具体的端口实现提供,以支持不同的串行设备。

本文地址:https://www.vps345.com/5405.html

搜索文章

Tags

PV计算 带宽计算 流量带宽 服务器带宽 上行带宽 上行速率 什么是上行带宽? CC攻击 攻击怎么办 流量攻击 DDOS攻击 服务器被攻击怎么办 源IP 服务器 linux 运维 游戏 云计算 javascript 前端 chrome edge ubuntu ssh deepseek Ollama 模型联网 API CherryStudio 进程 操作系统 进程控制 Ubuntu python MCP 阿里云 网络 网络安全 网络协议 llama 算法 opencv 自然语言处理 神经网络 语言模型 harmonyos 华为 开发语言 typescript 计算机网络 数据库 centos oracle 关系型 安全 分布式 github 创意 社区 flutter Hyper-V WinRM TrustedHosts RTSP xop RTP RTSPServer 推流 视频 udp unity react.js 前端面试题 node.js 持续部署 ssl java ollama ai 人工智能 llm php android Dell R750XS 科技 个人开发 pycharm ide pytorch macos adb 深度学习 YOLO 目标检测 计算机视觉 tcp/ip fastapi mcp mcp-proxy mcp-inspector fastapi-mcp agent sse nginx 监控 自动化运维 rust http django flask web3.py numpy 笔记 C 环境变量 进程地址空间 开源 Cline web安全 Kali Linux 黑客 渗透测试 信息收集 ollama下载加速 大模型 tomcat apache Deepseek kubernetes VMware安装Ubuntu Ubuntu安装k8s k8s mysql离线安装 ubuntu22.04 mysql8.0 机器学习 windows 搜索引擎 docker Qwen2.5-coder 离线部署 前端框架 运维开发 云原生 c语言 自动化 c# 低代码 c++ Flask FastAPI Waitress Gunicorn uWSGI Uvicorn prometheus jmeter 软件测试 vim 后端 kvm 无桌面 命令行 经验分享 媒体 微信公众平台 sqlserver YOLOv8 NPU Atlas800 A300I pro asi_bench DigitalOcean GPU服务器购买 GPU服务器哪里有 GPU服务器 matlab vscode mount挂载磁盘 wrong fs type LVM挂载磁盘 Centos7.9 ecm bpm elasticsearch jenkins qt stm32项目 单片机 stm32 AI 爬虫 数据集 ddos android studio 交互 websocket zotero WebDAV 同步失败 代理模式 ansible playbook es jvm AI编程 嵌入式 linux驱动开发 arm开发 嵌入式硬件 代码调试 ipdb 智能路由器 dell服务器 go asm golang IIS .net core Hosting Bundle .NET Framework vs2022 html 集成学习 集成测试 面试 性能优化 jdk intellij-idea 架构 gitee spring boot 串口服务器 oceanbase rc.local 开机自启 systemd 麒麟 chatgpt llama3 Chatglm 开源大模型 conda kylin 深度优先 图论 并集查找 换根法 树上倍增 宝塔面板访问不了 宝塔面板网站访问不了 宝塔面板怎么配置网站能访问 宝塔面板配置ip访问 宝塔面板配置域名访问教程 宝塔面板配置教程 容器 ESP32 java-ee Docker Compose docker compose docker-compose vue.js audio vue音乐播放器 vue播放音频文件 Audio音频播放器自定义样式 播放暂停进度条音量调节快进快退 自定义audio覆盖默认样式 nuxt3 vue3 bash 银河麒麟服务器操作系统 系统激活 sql KingBase rabbitmq uni-app 银河麒麟 kylin v10 麒麟 v10 postman mock mock server 模拟服务器 mock服务器 Postman内置变量 Postman随机数据 LDAP mysql 系统开发 binder 车载系统 framework 源码环境 filezilla 无法连接服务器 连接被服务器拒绝 vsftpd 331/530 HCIE 数通 maven intellij idea 腾讯云 实时音视频 dubbo ffmpeg 音视频 gateway Clion Nova ResharperC++引擎 Centos7 远程开发 外网访问 内网穿透 端口映射 pillow kamailio sip VoIP YOLOv12 温湿度数据上传到服务器 Arduino HTTP 统信 国产操作系统 虚拟机安装 .net 博客 openEuler 学习方法 程序人生 gitlab mongodb spring 大模型微调 蓝耘科技 元生代平台工作流 ComfyUI 多线程服务器 Linux网络编程 springsecurity6 oauth2 授权服务器 token sas DeepSeek-R1 API接口 live555 rtsp rtp json html5 firefox visualstudio WSL win11 无法解析服务器的名称或地址 .netcore EasyConnect zabbix 系统架构 Linux ecmascript nextjs react reactjs alias unalias 别名 unix 网络工程师 华为认证 RustDesk自建服务器 rustdesk服务器 docker rustdesk 黑客技术 ESXi https 流式接口 URL tcpdump 本地部署 api outlook DeepSeek 服务器繁忙 pyqt 小程序 微信小程序域名配置 微信小程序服务器域名 微信小程序合法域名 小程序配置业务域名 微信小程序需要域名吗 微信小程序添加域名 学习 Kylin-Server 服务器安装 机器人 gpu算力 软件工程 ux 多线程 SSE open Euler dde deepin 统信UOS 远程工作 LLM Web APP Streamlit hadoop W5500 OLED u8g2 TCP服务器 chfs ubuntu 16.04 网工 微信 支付 鸿蒙 微信支付 开放平台 opensearch helm ssrf 失效的访问控制 安装教程 GPU环境配置 Ubuntu22 CUDA PyTorch Anaconda安装 HTML audio 控件组件 vue3 audio音乐播放器 Audio标签自定义样式默认 vue3播放音频文件音效音乐 自定义audio播放器样式 播放暂停调整声音大小下载文件 MI300x 僵尸进程 AIGC openwrt sysctl.conf vm.nr_hugepages adobe elk Python 网络编程 聊天服务器 套接字 TCP 客户端 Socket hive Hive环境搭建 hive3环境 Hive远程模式 目标跟踪 OpenVINO 推理应用 svn xrdp 远程桌面 远程连接 centos-root /dev/mapper yum clean all df -h / du -sh string模拟实现 深拷贝 浅拷贝 经典的string类问题 三个swap 游戏服务器 TrinityCore 魔兽世界 开发环境 SSL证书 京东云 NFS redhat flash-attention 报错 Dify 群晖 文件分享 职场和发展 测试工具 自动化测试 性能测试 功能测试 odoo 服务器动作 Server action NPS 雨云服务器 雨云 缓存 能力提升 面试宝典 技术 IT信息化 游戏程序 崖山数据库 YashanDB 视频编解码 pip eureka 源码剖析 rtsp实现步骤 流媒体开发 Ubuntu 24.04.1 轻量级服务器 直播推流 cpu 内存 实时 使用 高效日志打印 串口通信日志 服务器日志 系统状态监控日志 异常记录日志 毕设 vmware 卡死 jar JAVA Java spring cloud 电脑 相差8小时 UTC 时间 redis netty 远程控制 远程看看 远程协助 rpc 远程过程调用 Windows环境 服务器数据恢复 数据恢复 存储数据恢复 北亚数据恢复 oracle数据恢复 aws 中间件 C语言 Ubuntu Server Ubuntu 22.04.5 ipython 物联网 硬件工程 三级等保 服务器审计日志备份 多进程 1024程序员节 FTP服务器 XCC Lenovo yum 飞牛NAS 飞牛OS MacBook Pro minio git gitea risc-v 软件需求 embedding 计算机 干货分享 黑客工具 密码爆破 bootstrap web Open WebUI ci/cd microsoft 软考 华为云 jupyter list 数据结构 Reactor 设计模式 C++ css skynet 联想开天P90Z装win10 Invalid Host allowedHosts vue 安全架构 多个客户端访问 IO多路复用 回显服务器 TCP相关API 硬件架构 LORA 大语言模型 NLP bonding 链路聚合 压力测试 r语言 tailscale derp derper 中转 矩阵 线性代数 电商平台 C++软件实战问题排查经验分享 0xfeeefeee 0xcdcdcdcd 动态库加载失败 程序启动失败 程序运行权限 标准用户权限与管理员权限 压测 ECS 宕机切换 服务器宕机 Portainer搭建 Portainer使用 Portainer使用详解 Portainer详解 Portainer portainer virtualenv webrtc 课程设计 大数据 debian cursor MCP server C/S LLM windows日志 transformer 数据挖掘 c Minecraft prompt easyui AI大模型 langchain DOIT 四博智联 Unity Dedicated Server Host Client 无头主机 负载均衡 宝塔面板 部署 服务器主板 AI芯片 信息与通信 Linux awk awk函数 awk结构 awk内置变量 awk参数 awk脚本 awk详解 H3C iDRAC R720xd 命名管道 客户端与服务端通信 freebsd PVE ios agi thingsboard postgresql XFS xfs文件系统损坏 I_O error 消息队列 shell 磁盘监控 iot cnn GoogLeNet 前后端分离 服务器无法访问 ip地址无法访问 无法访问宝塔面板 宝塔面板打不开 云服务 文件系统 路径解析 视觉检测 arm FunASR ASR MacOS录屏软件 佛山戴尔服务器维修 佛山三水服务器维修 file server http server web server VMware创建虚拟机 5G 3GPP 卫星通信 X11 Xming tidb GLIBC 进程信号 服务器配置 生物信息学 excel Wi-Fi DNS Spring Security rdp 实验 fpga开发 王者荣耀 交叉编译 交换机 硬件 设备 GPU PCI-Express 企业微信 Linux24.04 jetty undertow express word图片自动上传 word一键转存 复制word图片 复制word图文 复制word公式 粘贴word图文 粘贴word公式 UOS 统信操作系统 医疗APP开发 app开发 Agent 技能大赛 ISO镜像作为本地源 GaN HEMT 氮化镓 单粒子烧毁 辐射损伤 辐照效应 云电竞 云电脑 todesk 架构与原理 WebUI DeepSeek V3 Erlang OTP gen_server 热代码交换 事务语义 linux环境变量 MNN Qwen ip ui 备份SQL Server数据库 数据库备份 傲梅企业备份网络版 权限 音乐服务器 Navidrome 音流 npm ping++ 智能手机 NAS Termux Samba 银河麒麟桌面操作系统 Kylin OS 国产化 DeepSeek行业应用 Heroku 网站部署 xss camera Arduino 电子信息 pppoe radius IDEA hugo protobuf 序列化和反序列化 安装 SWAT 配置文件 服务管理 网络共享 gaussdb ruoyi vue-i18n 国际化多语言 vue2中英文切换详细教程 如何动态加载i18n语言包 把语言json放到服务器调用 前端调用api获取语言配置文件 saltstack 测试用例 kind AI写作 AI作画 next.js 部署next.js QQ 聊天室 数据库架构 数据管理 数据治理 数据编织 数据虚拟化 ocr 命令 mac AI agent idm 思科模拟器 思科 Cisco 算力 googlecloud qt项目 qt项目实战 qt教程 nac 802.1 portal muduo 微服务 国标28181 视频监控 监控接入 语音广播 流程 SIP SDP 其他 TRAE xml clickhouse firewalld 社交电子 计算机外设 EMQX MQTT 通信协议 弹性计算 虚拟化 KVM 计算虚拟化 弹性裸金属 midjourney windwos防火墙 defender防火墙 win防火墙白名单 防火墙白名单效果 防火墙只允许指定应用上网 防火墙允许指定上网其它禁止 根服务器 状态管理的 UDP 服务器 Arduino RTOS 漏洞 wsl2 wsl 重启 排查 系统重启 日志 原因 同步 备份 建站 驱动开发 mcu 安全威胁分析 阻塞队列 生产者消费者模型 服务器崩坏原因 ceph vscode 1.86 laravel Linux无人智慧超市 LInux多线程服务器 QT项目 LInux项目 单片机项目 VMware安装mocOS VMware macOS系统安装 grafana 直流充电桩 充电桩 IPMI junit 豆瓣 追剧助手 迅雷 nas 系统安全 云服务器 裸金属服务器 弹性裸金属服务器 p2p unity3d dify 深度求索 私域 知识库 网络穿透 Nuxt.js selete 高级IO SSH Xterminal 技术共享 bug 备选 网站 调用 示例 AD域 反向代理 致远OA OA服务器 服务器磁盘扩容 mybatis okhttp CORS 跨域 游戏机 Netty 即时通信 NIO dns 微信分享 Image wxopensdk 执法记录仪 智能安全帽 smarteye HTTP 服务器控制 ESP32 DeepSeek 产品经理 SysBench 基准测试 vasp安装 边缘计算 智能硬件 AutoDL IIS服务器 IIS性能 日志监控 编辑器 Linux PID Windsurf kafka 程序员 华为od MacMini Mac 迷你主机 mini Apple sqlite 宠物 毕业设计 免费学习 宠物领养 宠物平台 MS Materials openssl 密码学 业界资讯 模拟退火算法 WSL2 Dell HPE 联想 浪潮 code-server mosquitto 监控k8s集群 集群内prometheus 数据可视化 数据分析 gradle 华为机试 数据库系统 AISphereButler hibernate 大数据平台 银河麒麟高级服务器 外接硬盘 Kylin flink 框架搭建 HarmonyOS Next UDP的API使用 Docker Hub docker pull 镜像源 daemon.json 大模型入门 大模型教程 VR手套 数据手套 动捕手套 动捕数据手套 remote-ssh 剧本 ukui 麒麟kylinos openeuler rust腐蚀 7z 微信小程序 输入法 av1 电视盒子 机顶盒ROM 魔百盒刷机 springboot ftp 火绒安全 3d 数学建模 VPS 工业4.0 网络结构图 wireshark 显示过滤器 ICMP Wireshark安装 需求分析 规格说明书 vscode1.86 1.86版本 ssh远程连接 远程登录 telnet vSphere vCenter 软件定义数据中心 sddc RTMP 应用层 big data 远程 执行 sshpass 操作 minicom 串口调试工具 飞书 孤岛惊魂4 WebRTC gpt uniapp 网站搭建 serv00 恒源云 tcp echarts chrome devtools selenium chromedriver 传统数据库升级 银行 LLMs 单一职责原则 IPMITOOL BMC 硬件管理 opcua opcda KEPServer安装 oneapi open webui 向日葵 pdf asp.net大文件上传 asp.net大文件上传下载 asp.net大文件上传源码 ASP.NET断点续传 asp.net上传文件夹 asp.net上传大文件 .net core断点续传 Cursor iis CPU 主板 电源 网卡 VSCode 移动云 可信计算技术 鲲鹏 FTP 服务器 Headless Linux 服务器部署ai模型 语法 SSL 域名 rsyslog Anolis nginx安装 环境安装 linux插件下载 micropython esp32 mqtt raid5数据恢复 磁盘阵列数据恢复 自定义客户端 SAS 僵尸世界大战 游戏服务器搭建 银河麒麟操作系统 半虚拟化 硬件虚拟化 Hypervisor zookeeper 嵌入式实习 田俊楠 策略模式 单例模式 系统 黑苹果 虚拟机 pygame 小游戏 五子棋 v10 软件 armbian u-boot Trae IDE AI 原生集成开发环境 Trae AI 显卡驱动 内网环境 c/c++ 串口 大模型面经 大模型学习 ABAP AnythingLLM AnythingLLM安装 h.264 办公自动化 自动化生成 pdf教程 DevEco Studio cuda cudnn nvidia 项目部署到linux服务器 项目部署过程 prometheus数据采集 prometheus数据模型 prometheus特点 ip命令 新增网卡 新增IP 启动网卡 相机 金融 Windows 网卡的名称修改 eth0 ens33 用户缓冲区 源码 模拟实现 容器技术 gcc g++ g++13 cpp-httplib devops 混合开发 JDK regedit 开机启动 etcd 数据安全 RBAC fd 文件描述符 win服务器架设 windows server Ubuntu DeepSeek DeepSeek Ubuntu DeepSeek 本地部署 DeepSeek 知识库 DeepSeek 私有化知识库 本地部署 DeepSeek DeepSeek 私有化部署 基础入门 编程 firewall SSH 服务 SSH Server OpenSSH Server eNSP 企业网络规划 华为eNSP 网络规划 webgl rustdesk cocoapods xcode pyautogui threejs 3D SRS 流媒体 直播 考研 onlyoffice 在线office SenseVoice springboot远程调试 java项目远程debug docker远程debug java项目远程调试 springboot远程 python3.11 版本 dash 正则表达式 USB网络共享 yolov8 chrome 浏览器下载 chrome 下载安装 谷歌浏览器下载 私有化 环境迁移 OD机试真题 华为OD机试真题 服务器能耗统计 MySql UOS1070e lio-sam SLAM 智能音箱 智能家居 浏览器开发 AI浏览器 cd 目录切换 ssh漏洞 ssh9.9p2 CVE-2025-23419 sentinel epoll visual studio code 知识图谱 dba Ubuntu22.04 开发人员主页 trea idea tensorflow 蓝桥杯 政务 分布式系统 监控运维 Prometheus Grafana swoole 移动魔百盒 USB转串口 CH340 CLion Node-Red 编程工具 流编程 harmonyOS面试题 繁忙 解决办法 替代网站 汇总推荐 AI推理 etl CDN apt 国内源 邮件APP 免费软件 网络攻击模型 wsgiref Web 服务器网关接口 curl wget 端口 查看 ss 加解密 Yakit yaklang gpt-3 文心一言 Docker引擎已经停止 Docker无法使用 WSL进度一直是0 镜像加速地址 ardunio BLE 无人机 deepseek r1 Cookie 影刀 #影刀RPA# Kali triton 模型分析 网络用户购物行为分析可视化平台 大数据毕业设计 大文件分片上传断点续传及进度条 如何批量上传超大文件并显示进度 axios大文件切片上传详细教 node服务器合并切片 vue3大文件上传报错提示错误 大文件秒传跨域报错cors 键盘 流量运营 RoboVLM 通用机器人策略 VLA设计哲学 vlm fot robot 视觉语言动作模型 具身智能 序列化反序列化 链表 阿里云ECS iphone 代理服务器 ue4 着色器 ue5 虚幻 openstack Xen iftop 网络流量监控 游戏引擎 实习 线程 TCP协议 make命令 makefile文件 开机自启动 图形化界面 粘包问题 rag ragflow ragflow 源码启动 防火墙 NAT转发 NAT Server 端口测试 docker搭建nacos详解 docker部署nacos docker安装nacos 腾讯云搭建nacos centos7搭建nacos 飞牛nas fnos 鸿蒙系统 uv glibc 安卓 P2P HDLC 常用命令 文本命令 目录命令 figma 镜像 springcloud QT 5.12.12 QT开发环境 Ubuntu18.04 Google pay Apple pay 金仓数据库 2025 征文 数据库平替用金仓 ROS 自动驾驶 程序 性能分析 Unity插件 RAID RAID技术 磁盘 存储 iventoy VmWare OpenEuler css3 OpenManus navicat Jellyfin wps db mamba Vmamba mq rocketmq TrueLicense Typore less 实时互动 免费域名 域名解析 信号处理 超融合 linux安装配置 我的世界 我的世界联机 数码 RAGFLOW RAG 检索增强生成 文档解析 大模型垂直应用 rnn bat UDP 大模型应用 sqlite3 OpenSSH 我的世界服务器搭建 ruby docker run 数据卷挂载 交互模式 串口驱动 CH341 uart 485 IPv4 子网掩码 公网IP 私有IP Python基础 Python教程 Python技巧 bcompare Beyond Compare 模拟器 教程 带外管理 linux 命令 sed 命令 RAGFlow 本地知识库部署 DeepSeek R1 模型 seatunnel grub 版本升级 扩容 服务器时间 昇腾 npu 设置代理 实用教程 游戏开发 wordpress 无法访问wordpess后台 打开网站页面错乱 linux宝塔面板 wordpress更换服务器 GCC Linux环境 C# MQTTS 双向认证 emqx iperf3 带宽测试 环境配置 自动化任务管理 Claude VLAN 企业网络 anaconda 在线预览 xlsx xls文件 在浏览器直接打开解析xls表格 前端实现vue3打开excel 文件地址url或接口文档流二进 基础环境 EMUI 回退 降级 升级 k8s资源监控 annotations自动化 自动化监控 监控service 监控jvm log4j KylinV10 麒麟操作系统 Vmware qemu libvirt kali 共享文件夹 webstorm dns是什么 如何设置电脑dns dns应该如何设置 并查集 leetcode 强制清理 强制删除 mac废纸篓 Xinference vr aarch64 编译安装 HPC can 线程池 中兴光猫 换光猫 网络桥接 自己换光猫 Deepseek-R1 私有化部署 推理模型 EtherCAT转Modbus ECT转Modbus协议 EtherCAT转485网关 ECT转Modbus串口网关 EtherCAT转485协议 ECT转Modbus网关 IM即时通讯 剪切板对通 HTML FORMAT iBMC UltraISO ArkUI 多端开发 智慧分发 应用生态 鸿蒙OS lua Playwright Radius 树莓派 VNC ssh远程登录 Ark-TS语言 单元测试 个人博客 linux上传下载 健康医疗 互联网医院 域名服务 DHCP 符号链接 配置 灵办AI HiCar CarLife+ CarPlay QT RK3588 音乐库 飞牛 换源 Debian 高效远程协作 TrustViewer体验 跨设备操作便利 智能远程控制 HAProxy perl 虚拟显示器 edge浏览器 cmos 元服务 应用上架 物联网开发 ros 小番茄C盘清理 便捷易用C盘清理工具 小番茄C盘清理的优势尽显何处? 教你深度体验小番茄C盘清理 C盘变红?!不知所措? C盘瘦身后电脑会发生什么变化? yaml Ultralytics 可视化 显示管理器 lightdm gdm trae AI代码编辑器 crosstool-ng SEO arcgis deekseek Ubuntu共享文件夹 共享目录 Linux共享文件夹 mariadb 多层架构 解耦 流水线 脚本式流水线 yum源切换 更换国内yum源 腾讯云大模型知识引擎 efficientVIT YOLOv8替换主干网络 TOLOv8 bot Docker 图像处理 Linux find grep 分析解读 eclipse jina CrewAI 匿名管道 代理 rime frp perf LInux Linux的基础指令 信号 查询数据库服务IP地址 SQL Server 语音识别 composer 产测工具框架 IMX6ULL 管理框架 本地部署AI大模型 迁移指南 react native Logstash 日志采集 llama.cpp 网络药理学 生信 gromacs 分子动力学模拟 MD 动力学模拟 开发 小艺 Pura X EtherNet/IP串口网关 EIP转RS485 EIP转Modbus EtherNet/IP网关协议 EIP转RS485网关 EIP串口服务器 milvus 双系统 SVN Server tortoise svn MVS 海康威视相机 .net mvc断点续传 xpath定位元素 信息可视化 网页设计 ubuntu24.04.1 ros2 moveit 机器人运动 uni-file-picker 拍摄从相册选择 uni.uploadFile H5上传图片 微信小程序上传图片 proxy模式 程序员创富 烟花代码 烟花 元旦 状态模式 IO模型 强化学习 做raid 装系统 Java Applet URL操作 服务器建立 Socket编程 网络文件读取 服务器管理 配置教程 网站管理 fast 软链接 硬链接 VS Code HarmonyOS n8n 工作流 workflow CVE-2024-7347 内网服务器 内网代理 内网通信 VM搭建win2012 win2012应急响应靶机搭建 攻击者获取服务器权限 上传wakaung病毒 应急响应并溯源 挖矿病毒处置 应急响应综合性靶场 dity make rtsp服务器 rtsp server android rtsp服务 安卓rtsp服务器 移动端rtsp服务 大牛直播SDK autodl 微信开放平台 微信公众号配置 hexo keepalived sonoma 自动更新 实战案例 小智AI服务端 xiaozhi TTS 主从复制 DenseNet web3 AD 域管理 SSH 密钥生成 SSH 公钥 私钥 生成 rclone AList webdav fnOS 抗锯齿 IMX317 MIPI H265 VCU ArcTS 登录 ArcUI GridItem 磁盘镜像 服务器镜像 服务器实时复制 实时文件备份 ShenTong arkUI 服务网格 istio docker搭建pg docker搭建pgsql pg授权 postgresql使用 postgresql搭建 IMM matplotlib wpf docker命令大全 毕昇JDK 合成模型 扩散模型 图像生成 GRUB引导 Linux技巧 ubuntu24 vivado24 spark HistoryServer Spark YARN jobhistory minecraft nfs k8s集群资源管理 云原生开发 safari 历史版本 下载 OpenHarmony 真机调试 怎么卸载MySQL MySQL怎么卸载干净 MySQL卸载重新安装教程 MySQL5.7卸载 Linux卸载MySQL8.0 如何卸载MySQL教程 MySQL卸载与安装 分布式训练 AP配网 AK配网 小程序AP配网和AK配网教程 WIFI设备配网小程序UDP开 捆绑 链接 谷歌浏览器 youtube google gmail 大大通 第三代半导体 碳化硅 图形渲染 ai工具 java-rocketmq 虚拟局域网 Ubuntu 24 常用命令 Ubuntu 24 Ubuntu vi 异常处理 ldap AI-native Docker Desktop sdkman sequoiaDB pgpool GIS 遥感 WebGIS lsb_release /etc/issue /proc/version uname -r 查看ubuntu版本 CentOS 离线部署dify 存储维护 NetApp存储 EMC存储 PX4 MDK 嵌入式开发工具 论文笔记 sublime text PPI String Cytoscape CytoHubba seleium 本地化部署 推荐算法 代码 对比 工具 meld DiffMerge 渗透 玩机技巧 软件分享 软件图标 kernel 宝塔 西门子PLC 通讯 网络建设与运维 软负载 deployment daemonset statefulset cronjob visual studio AI Agent 字节智能运维 deep learning 人工智能生成内容 ranger MySQL8.0 上传视频至服务器代码 vue3批量上传多个视频并预览 如何实现将本地视频上传到网页 element plu视频上传 ant design vue vue3本地上传视频及预览移除 nlp 拓扑图 嵌入式系统开发 钉钉 conda配置 conda镜像源 DeepSeek r1 远程服务 x64 SIGSEGV xmm0 稳定性 看门狗 DBeaver miniapp 调试 debug 断点 网络API请求调试方法 Attention 大模型部署 信创 信创终端 中科方德 查看显卡进程 fuser ArtTS 搭建个人相关服务器 docker部署翻译组件 docker部署deepl docker搭建deepl java对接deepl 翻译组件使用 大模型推理 嵌入式Linux IPC MQTT协议 消息服务器 网络爬虫 gnu ubuntu20.04 开机黑屏 沙盒 word mm-wiki搭建 linux搭建mm-wiki mm-wiki搭建与使用 mm-wiki使用 mm-wiki详解 多路转接 欧标 OCPP hosts 论文阅读 Linux的权限 监控k8s 监控kubernetes 运维监控 rpa HarmonyOS NEXT 原生鸿蒙 yum换源 网络搭建 神州数码 神州数码云平台 云平台 数据仓库 kerberos fstab neo4j WebVM 增强现实 沉浸式体验 应用场景 技术实现 案例分析 AR 风扇控制软件 虚幻引擎 问题解决 智能电视 李心怡 自动化编程 CentOS Stream 聚类 性能调优 安全代理 ai小智 语音助手 ai小智配网 ai小智教程 esp32语音助手 diy语音助手 docker部署Python 热榜 searxng xshell termius iterm2 数据库开发 database 服务器正确解析请求体 Redis Desktop IO js 软件卸载 系统清理 DocFlow Qwen2.5-VL vllm 项目部署 鸿蒙开发 移动开发 dock 加速