0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

串口驱动分析之serial driver

嵌入式与Linux那些事 来源:嵌入式与Linux那些事 2024-09-04 14:23 次阅读

简介

前两节我们介绍串口驱动的框架和tty core部分。这节我们介绍和硬件紧密相关的串口驱动部分。

UART驱动部分依赖于硬件平台,而TTY驱动和具体的平台无关。虽然UART部分依赖于平台,但是不管是哪个硬件平台,驱动的思路都是一致的,下面分模块来分别介绍。

关键数据结构

struct uart_driver

struct uart_driver结构体本身并不包含底层UART硬件的操作方法,其是所有串口设备驱动的抽象和封装。起到了连接硬件设备驱动和TTY驱动的作用。注册了struct uart_driver后还不能使用UART设备,还需要关联具体的UART设备。

uart_driver 结构体表示 UART 驱动, 它定义在include/linux/serial_core.h文件中,内容如下:

structuart_driver{
structmodule*owner;
constchar*driver_name;
constchar*dev_name;
intmajor;
intminor;
intnr;
structconsole*cons;

/*
*theseareprivate;thelowleveldrivershouldnot
*touchthese;theyshouldbeinitialisedtoNULL
*/
structuart_state*state;
structtty_driver*tty_driver;
};

owner:指向该驱动程序的拥有者模块的指针,即加载该驱动程序的内核模块。

driver_name:字符串,表示驱动程序的名称。

dev_name:字符串,表示设备名称,即驱动程序控制的设备文件的名称,比如ttyS。

major:表示设备文件的主设备号。

minor:表示设备文件的次设备号。

nr:整数,表示该驱动程序控制的设备数量。

cons:指向 struct console 类型的指针,表示该串口设备所绑定的控制台。

此外,结构体中还包含了两个私有的指针字段:

state:指向 struct uart_state 类型的指针,表示该驱动程序内部的状态信息

tty_driver:指向 struct tty_driver 类型的指针,表示该驱动程序所对应的 tty 驱动程序。

struct uart_port

一个串口芯片上往往有多个串行端口(serial ports,对应于一个物理上的串口),这些串行端口具备相同的操作机制。Linux内核将这些串行端口用struct uart_port结构体描述。struct uart_port用于描述一个UART端口的中断、I/O内存地址、FIFO大小、端口类型等信息。

在 Linux 内核中,每个串口设备都会对应一个 struct uart_port 数据结构,并且这个数据结构会作为串口设备的一个属性被保存在相应的设备节点中。

当应用程序通过打开设备节点来访问串口设备时,内核会通过设备节点获取对应的 struct uart_port 数据结构,然后通过这个数据结构来进行串口的读写等操作。

structuart_port{
spinlock_tlock;/*portlock*/
unsignedlongiobase;/*in/out[bwl]*/
unsignedchar__iomem*membase;/*read/write[bwl]*/
unsignedint(*serial_in)(structuart_port*,int);
void(*serial_out)(structuart_port*,int,int);
void(*set_termios)(structuart_port*,
structktermios*new,
structktermios*old);
void(*set_mctrl)(structuart_port*,unsignedint);
int(*startup)(structuart_port*port);
void(*shutdown)(structuart_port*port);
void(*throttle)(structuart_port*port);
void(*unthrottle)(structuart_port*port);
int(*handle_irq)(structuart_port*);
void(*pm)(structuart_port*,unsignedintstate,
unsignedintold);
void(*handle_break)(structuart_port*);
int(*rs485_config)(structuart_port*,
structserial_rs485*rs485);
unsignedintirq;/*irqnumber*/
unsignedlongirqflags;/*irqflags*/
unsignedintuartclk;/*baseuartclock*/
unsignedintfifosize;/*txfifosize*/
unsignedcharx_char;/*xon/xoffchar*/
unsignedcharregshift;/*regoffsetshift*/
unsignedchariotype;/*ioaccessstyle*/
unsignedcharunused1;

unsignedintread_status_mask;/*driverspecific*/
unsignedintignore_status_mask;/*driverspecific*/
structuart_state*state;/*pointertoparentstate*/
structuart_icounticount;/*statistics*/
structconsole*cons;/*structconsole,ifany*/
/*flagsmustbeupdatedwhileholdingportmutex*/
upf_tflags;
/*
*Mustholdtermios_rwsem,portmutexandportlocktochange;
*canholdanyonelocktoread.
*/
upstat_tstatus;

inthw_stopped;/*sw-assistedCTSflowstate*/
unsignedintmctrl;/*currentmodemctrlsettings*/
unsignedinttimeout;/*character-basedtimeout*/
unsignedinttype;/*porttype*/
conststructuart_ops*ops;
unsignedintcustom_divisor;
unsignedintline;/*portindex*/
unsignedintminor;
resource_size_tmapbase;/*forioremap*/
resource_size_tmapsize;
structdevice*dev;/*parentdevice*/
unsignedcharhub6;/*thisshouldbeinthe8250driver*/
unsignedcharsuspended;
unsignedcharirq_wake;
unsignedcharunused[2];
structattribute_group*attr_group;/*portspecificattributes*/
conststructattribute_group**tty_groups;/*allattributes(serialcoreuseonly)*/
structserial_rs485rs485;
void*private_data;/*genericplatformdatapointer*/
};

unsigned long iobase: 指定了该串口设备在I/O空间中的基地址。

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): 函数指针,用于设置串口设备的终端参数。

void (*set_mctrl)(struct uart_port *, unsigned int): 函数指针,用于设置串口设备的 modem 控制信号

int (*startup)(struct uart_port *port): 函数指针,用于初始化串口设备并启动传输。

void (*shutdown)(struct uart_port *port): 函数指针,用于关闭串口设备。

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: I/O 访问类型。

unsigned char unused1: 未使用的成员变量。

unsigned int read_status_mask: 用于指定读取状态的屏蔽位。

unsigned int ignore_status_mask: 用于指定忽略状态的屏蔽位。

struct uart_state *state: 指向该串口设备所在状态结构体的指针。

struct uart_icount icount: 用于存储串口设备的统计信息。

struct console *cons: 指向该串口设备所属控制台设备的指针。

unsigned int mctrl:当前调制解调器控制(Modem Control)的设置。这个值包含了当前控制信号(如DTR、RTS、DSR、CTS等)的状态。通常由硬件控制。

unsigned int timeout:基于字符的超时时间。当字符被传输到UART端口时,如果在规定的时间内没有收到下一个字符,则会超时并发送通知。通常由驱动程序设置。

unsigned int type:端口类型。这个值通常用于标识UART硬件的特殊性质(如芯片类型、波特率范围等)。

const struct uart_ops *ops:一个指向struct uart_ops结构体的指针。这个结构体包含了与UART驱动程序相关的函数指针,如UART读、写、启动、停止等等。

unsigned int custom_divisor:自定义除数,用于实现非标准波特率。这个值通常由驱动程序设置。

unsigned int line:端口索引,用于标识该UART端口的编号。

unsigned int minor:端口的次设备号,用于标识该UART端口在系统中的位置。

resource_size_t mapbase、resource_size_t mapsize:映射区域的起始地址和大小。这些值通常由驱动程序设置,用于将UART端口的物理地址映射到虚拟地址。

struct device *dev:指向父设备的指针。通常是该UART设备所连接的总线控制器设备。

unsigned char hub6:用于指示Hub6电路板的状态。这个变量应该是在8250驱动程序中定义的。

unsigned char suspended:用于指示该端口是否被挂起。

unsigned char irq_wake:用于指示该端口是否支持唤醒中断。

unsigned char unused[2]:未使用的字节。

struct attribute_group *attr_group:指向属性组的指针。属性组包含了UART设备的属性和操作,如设备状态、波特率设置等等。

const struct attribute_group **tty_groups:指向指针数组的指针,该数组包含了所有属性组的指针,供串行核心使用。

struct serial_rs485 rs485:RS485配置结构体,用于RS485通信。

void *private_data:指向私有数据的指针。这个指针通常由驱动程序使用,用于保存驱动程序特定的数据。

struct uart_ops

Linux 系统收发数据最终调用的都是 ops 中的函数。 ops 是 uart_ops类型的结构体指针变量。uart硬件操作函数集合,底层硬件驱动必须实现这个结构体。

uart_ops结构体 用于定义一个串口驱动程序的接口,让上层调用这些接口实现串口的读写等操作。它包含了很多函数指针,每个函数指针对应了一个特定的串口操作。

在Linux内核中,串口的驱动程序是分为两层实现的:串口芯片驱动程序和 serial core 层。其中,serial core 层提供了大量的函数接口,供上层的串口芯片驱动程序使用,这些函数接口的定义就包含在了 struct uart_ops 结构体中。

当编写串口芯片驱动程序时,需要实现 struct uart_ops 结构体中定义的各个函数接口,以便 serial core 层调用。

例如,在芯片驱动程序中实现的 uart_start() 函数就对应了 struct uart_ops 结构体中的 startup 函数指针。

因此,struct uart_ops 结构体是串口驱动程序实现的关键,其定义了驱动程序需要实现的所有函数接口,并与 serial core 层进行了对接。

structuart_ops{
unsignedint(*tx_empty)(structuart_port*);
void(*set_mctrl)(structuart_port*,unsignedintmctrl);
unsignedint(*get_mctrl)(structuart_port*);
void(*stop_tx)(structuart_port*);
void(*start_tx)(structuart_port*);
void(*throttle)(structuart_port*);
void(*unthrottle)(structuart_port*);
void(*send_xchar)(structuart_port*,charch);
void(*stop_rx)(structuart_port*);
void(*enable_ms)(structuart_port*);
void(*break_ctl)(structuart_port*,intctl);
int(*startup)(structuart_port*);
void(*shutdown)(structuart_port*);
void(*flush_buffer)(structuart_port*);
void(*set_termios)(structuart_port*,structktermios*new,
structktermios*old);
void(*set_ldisc)(structuart_port*,structktermios*);
void(*pm)(structuart_port*,unsignedintstate,
unsignedintoldstate);
void(*wake_peer)(structuart_port*);

/*
*Returnastringdescribingthetypeoftheport
*/
constchar*(*type)(structuart_port*);

/*
*ReleaseIOandmemoryresourcesusedbytheport.
*Thisincludesiounmapifnecessary.
*/
void(*release_port)(structuart_port*);

/*
*RequestIOandmemoryresourcesusedbytheport.
*Thisincludesiomappingtheportifnecessary.
*/
int(*request_port)(structuart_port*);
void(*config_port)(structuart_port*,int);
int(*verify_port)(structuart_port*,structserial_struct*);
int(*ioctl)(structuart_port*,unsignedint,unsignedlong);
#ifdefCONFIG_CONSOLE_POLL
int(*poll_init)(structuart_port*);
void(*poll_put_char)(structuart_port*,unsignedchar);
int(*poll_get_char)(structuart_port*);
#endif
};

tx_empty():检查串口的发送缓冲区是否为空,用于判断是否可以发送数据。

set_mctrl():设置串口的 modem 控制信号,如 RTS、DTR 等。

get_mctrl():获取串口的 modem 控制信号。

stop_tx():停止当前正在发送的数据。

start_tx():开始发送数据。

throttle():限制发送速率,减少发送的数据量。

unthrottle():取消限制发送速率。

send_xchar():发送一个 XON 或 XOFF 字符,用于流控。

stop_rx():停止接收数据。

enable_ms():启用串口的 modem 状态检测功能。

break_ctl():发送一个 break 信号。

startup():初始化串口硬件。

shutdown():关闭串口硬件。

flush_buffer():清空串口的缓冲区。

set_termios():设置串口的终端参数。

set_ldisc():设置串口的行规则。

pm():实现串口的 power management。

wake_peer():用于唤醒其他休眠状态的串口。

另外,还包含了一些函数指针用于处理串口的 IO 资源:

type():返回描述串口类型的字符串。

release_port():释放串口的 IO 和内存资源,包括解除 IO 映射等。

request_port():请求串口的 IO 和内存资源,包括 IO 映射等。

config_port():配置串口的参数。

verify_port():验证串口的参数是否正确。

ioctl():实现串口设备的 ioctl 接口。

struct uart_state

uart_state 表示 UART 状态,并与 struct uart_port 结构体配合使用来管理 UART 端口。

struct uart_port 结构体表示 UART 端口的硬件信息和操作,而 struct uart_state 结构体则表示与该端口相关的软件状态。

由于 UART 状态可以包含多个,因此可以在同一时刻使用多个 UART 状态来管理多个 UART 端口的操作。

structuart_state{
structtty_portport;

enumuart_pm_statepm_state;
structcirc_bufxmit;

structuart_port*uart_port;
};

struct tty_port port:表示 tty 端口的状态信息,包括接受和发送缓冲区,控制信息和流控信息等等。

enum uart_pm_state pm_state:表示串口设备的电源管理状态,可以是 UART_PM_STATE_ON、UART_PM_STATE_OFF 或 UART_PM_STATE_UNDEFINED。

struct circ_buf xmit:表示串口设备的发送缓冲区,用于存储待发送的数据。

struct uart_port *uart_port:表示该串口设备对应的 struct uart_port 结构体。

当应用程序向串口设备写入数据时,数据将被存储到 xmit 缓冲区中,并且将触发串口驱动程序的数据发送处理函数。这个函数会从 xmit 缓冲区中取出数据,并通过 uart_port 中的函数指针将数据发送到物理串口。在发送数据时,驱动程序还会根据串口的流控状态进行数据流控制。

当收到数据时,数据将被存储到 port 的接受缓冲区中,并且将触发串口驱动程序的数据接收处理函数。处理函数将从接受缓冲区中取出数据并将其传递给应用程序。

数据结构抽象完毕后,serial core向下层的driver提供了方便的编程API,主要包括以下函数。

关键API

uart_register_driver

uart_register_driver将定义并填充好的uart driver注册到kernel中,一般在驱动模块的init接口中被调用。

intuart_register_driver(structuart_driver*drv)
{
structtty_driver*normal;
inti,retval;

BUG_ON(drv->state);

drv->state=kzalloc(sizeof(structuart_state)*drv->nr,GFP_KERNEL);
if(!drv->state)
gotoout;

normal=alloc_tty_driver(drv->nr);
if(!normal)
gotoout_kfree;

drv->tty_driver=normal;

normal->driver_name=drv->driver_name;
normal->name=drv->dev_name;
normal->major=drv->major;
normal->minor_start=drv->minor;
normal->type=TTY_DRIVER_TYPE_SERIAL;
normal->subtype=SERIAL_TYPE_NORMAL;
normal->init_termios=tty_std_termios;
normal->init_termios.c_cflag=B9600|CS8|CREAD|HUPCL|CLOCAL;
normal->init_termios.c_ispeed=normal->init_termios.c_ospeed=9600;
normal->flags=TTY_DRIVER_REAL_RAW|TTY_DRIVER_DYNAMIC_DEV;
normal->driver_state=drv;
tty_set_operations(normal,&uart_ops);

/*
*InitialisetheUARTstate(s).
*/
for(i=0;i< drv->nr;i++){
structuart_state*state=drv->state+i;
structtty_port*port=&state->port;

tty_port_init(port);
port->ops=&uart_port_ops;
}

retval=tty_register_driver(normal);
if(retval>=0)
returnretval;

for(i=0;i< drv->nr;i++)
tty_port_destroy(&drv->state[i].port);
put_tty_driver(normal);
out_kfree:
kfree(drv->state);
out:
return-ENOMEM;
}

uart_register_driver()注册所做工作如下:

根据driver支持的最大设备数,申请n个 uart_state 空间,每一个 uart_state 都有一个 uart_port 。

接着它会分配一个 tty_driver 对象,并初始化它的各个属性,如 driver_name,name,major,minor_start 等等。这些属性是用于在 TTY 子系统中创建 tty 设备的,它们的值来自于 uart_driver 对象中指定的值。

接下来,它会在 tty_driver 中设置 tty 操作,其中 tty_ops 是一个结构体,定义了 UART 串行接口所需要的函数。这些函数在串口设备注册后,当有数据进出串口时,TTY 子系统会调用这些函数。tty_set_operations() 函数用于在 tty_driver 中设置 tty 操作。

在初始化完 tty_driver 后,函数会遍历所有的 UART 设备状态对象,并初始化它们。这些状态对象被存储在 uart_driver 对象的 state 字段中。每个 UART 设备状态对象包含一个 tty_port 对象,其中存储了关于该串口设备的信息,例如流控、字长、奇偶校验等等。在此处, tty_port 的操作被设置为 uart_port_ops,它包含了具体实现 UART 串行接口所需的函数。

最后会调用 tty_register_driver() 函数来向内核注册 tty 驱动程序,并将驱动程序的 tty_driver 结构体与 uart_driver 结构体相关联。

如果注册失败,该函数将释放之前分配的内存。如果注册成功,该函数将返回 0,否则将返回一个负的错误码。

总结一句话:tty serial core底层驱动层和tty层之间的联系需要从uart_register_driver()中连接,tty_driver是在uart_driver注册过程中构建的。

uart_unregister_driver

uart_unregister_driver是一个Linux内核中的串口驱动反注册函数,用于将之前注册的驱动程序与系统中的串口设备取消关联。

/**
*uart_unregister_driver-removeadriverfromtheuartcorelayer
*@drv:lowleveldriverstructure
*
*Removeallreferencestoadriverfromthecoredriver.Thelow
*leveldrivermusthaveremovedallitsportsviathe
*uart_remove_one_port()ifitregisteredthemwithuart_add_one_port().
*(ie,drv->port==NULL)
*/
voiduart_unregister_driver(structuart_driver*drv)
{
structtty_driver*p=drv->tty_driver;
unsignedinti;
/*获取与该驱动程序关联的tty_driver实例*/
tty_unregister_driver(p);

/*取消注册驱动程序,将它与系统中的tty设备断开关联*/
put_tty_driver(p);

/*释放该tty_driver实例,如果此时该实例的使用计数为零,即没有其他模块在使用该实例,那么它将会被完全卸载并释放所有内存资源*/
for(i=0;i< drv->nr;i++)
tty_port_destroy(&drv->state[i].port);
kfree(drv->state);
drv->state=NULL;
drv->tty_driver=NULL;
}

uart_add_one_port

uart_add_one_port用于将一个UART端口添加到UART驱动程序的状态表中,并注册TTY端口设备,让用户空间能够通过该设备与UART通信。

/**
*uart_add_one_port-attachadriver-definedportstructure
*@drv:pointertotheuartlowleveldriverstructureforthisport
*@uport:uartportstructuretouseforthisport.
*
*Thisallowsthedrivertoregisteritsownuart_portstructure
*withthecoredriver.Themainpurposeistoallowthelow
*leveluartdriverstoexpanduart_port,ratherthanhavingyet
*morelevelsofstructures.
*/
intuart_add_one_port(structuart_driver*drv,structuart_port*uport)
{
structuart_state*state;
structtty_port*port;
intret=0;
structdevice*tty_dev;
intnum_groups;

/*检查是否在中断上下文中,如果是则直接返回错误*/
BUG_ON(in_interrupt());

/*检查所添加的端口是否超出驱动程序支持的范围,如果是则返回EINVAL*/
if(uport->line>=drv->nr)
return-EINVAL;

/*获取该端口所对应的状态信息(uart_state)以及端口(tty_port)*/
state=drv->state+uport->line;
port=&state->port;

mutex_lock(&port_mutex);
mutex_lock(&port->mutex);

/*检查端口是否已经被其他设备占用,如果是则返回EINVAL*/
if(state->uart_port){
ret=-EINVAL;
gotoout;
}

/*链接端口和驱动程序状态表,并进行相应的初始化工作,包括PM状态、控制台、spinlock等*/
state->uart_port=uport;
uport->state=state;

state->pm_state=UART_PM_STATE_UNDEFINED;
uport->cons=drv->cons;
uport->minor=drv->tty_driver->minor_start+uport->line;

/*
*Ifthisportisaconsole,thenthespinlockisalready
*initialised.
*/
if(!(uart_console(uport)&&(uport->cons->flags&CON_ENABLED))){
spin_lock_init(&uport->lock);
lockdep_set_class(&uport->lock,&port_lock_key);
}
if(uport->cons&&uport->dev)
of_console_check(uport->dev->of_node,uport->cons->name,uport->line);

/*配置端口的属性,例如波特率、数据位、停止位等*/
uart_configure_port(drv,state,uport);

num_groups=2;
if(uport->attr_group)
num_groups++;

/*分配并设置TTY设备属性组,这些属性组包括TTY设备通用属性组和用户自定义属性组*/
uport->tty_groups=kcalloc(num_groups,sizeof(*uport->tty_groups),
GFP_KERNEL);
if(!uport->tty_groups){
ret=-ENOMEM;
gotoout;
}
uport->tty_groups[0]=&tty_dev_attr_group;
if(uport->attr_group)
uport->tty_groups[1]=uport->attr_group;

/*注册TTY端口设备,并将其与tty_driver和tty_port关联起来*/
tty_dev=tty_port_register_device_attr(port,drv->tty_driver,
uport->line,uport->dev,port,uport->tty_groups);
/*如果注册成功,将该设备标记为可唤醒*/
if(likely(!IS_ERR(tty_dev))){
device_set_wakeup_capable(tty_dev,1);
}else{
dev_err(uport->dev,"Cannotregisterttydeviceonline%d
",
uport->line);
}

/*
*EnsureUPF_DEADisnotset.
*/
uport->flags&=~UPF_DEAD;

out:
mutex_unlock(&port->mutex);
mutex_unlock(&port_mutex);

returnret;
}

uart_remove_one_port

uart_remove_one_port用于从核心驱动程序中分离(断开)一个指定的端口结构。

/**
*uart_remove_one_port-detachadriverdefinedportstructure
*@drv:pointertotheuartlowleveldriverstructureforthisport
*@uport:uartportstructureforthisport
*
*Thisunhooks(andhangsup)thespecifiedportstructurefromthe
*coredriver.Nofurthercallswillbemadetothelow-levelcode
*forthisport.
*/
intuart_remove_one_port(structuart_driver*drv,structuart_port*uport)
{
structuart_state*state=drv->state+uport->line;
structtty_port*port=&state->port;
structtty_struct*tty;
intret=0;

/*检查当前是否处于中断上下文中*/
BUG_ON(in_interrupt());

/*检查uart状态结构中的uart端口指针是否等于传递给该函数的uart端口指针,如果不是则打印一条错误消息*/
if(state->uart_port!=uport)
dev_alert(uport->dev,"Removingwrongport:%p!=%p
",
state->uart_port,uport);

/*获取tty端口结构的互斥锁,该锁用于防止并发修改端口状态*/
mutex_lock(&port_mutex);

/*获取tty端口结构的互斥锁,然后检查uart端口指针是否为空。如果为空,则表示当前端口已被删除。在这种情况下,将返回-EINVAL并解锁互斥锁*/
mutex_lock(&port->mutex);
if(!state->uart_port){
mutex_unlock(&port->mutex);
ret=-EINVAL;
gotoout;
}
/*锁定port->mutex互斥锁,并将uport->flags设置为UPF_DEAD,表示该端口已经被关闭。之后解锁port->mutex。*/
uport->flags|=UPF_DEAD;
mutex_unlock(&port->mutex);

/*从tty层中删除设备*/
tty_unregister_device(drv->tty_driver,uport->line);

/*获取tty设备对应的tty结构体,并使用tty_vhangup()函数关闭该tty设备的控制终端。最后,使用tty_kref_put()函数释放tty结构体的引用计数。*/
tty=tty_port_tty_get(port);
if(tty){
tty_vhangup(port->tty);
tty_kref_put(tty);
}

/*如果该端口用作控制台,则使用unregister_console()函数取消该端口的控制台注册*/
if(uart_console(uport))
unregister_console(uport->cons);

/*根据uport->type的值来释放端口的IO和内存资源,如果uport->type的值为PORT_UNKNOWN,则表示没有对应的资源需要释放*/
if(uport->type!=PORT_UNKNOWN)
uport->ops->release_port(uport);
kfree(uport->tty_groups);

/*将uport->type的值设置为PORT_UNKNOWN,表示该端口不再存在。同时将state->uart_port设置为NULL,表示state对应的端口不再与uport相关联。*/
uport->type=PORT_UNKNOWN;

state->uart_port=NULL;
out:
mutex_unlock(&port_mutex);

returnret;
}

uart_write_wakeup

uart_write_wakeupuart_write_wakeup唤醒上层因向串口端口写数据而阻塞的进程,通常在串口发送中断处理函数中调用该函数。

/*
*Thisroutineisusedbytheinterrupthandlertoscheduleprocessingin
*thesoftwareinterruptportionofthedriver.
*/
voiduart_write_wakeup(structuart_port*port)
{
structuart_state*state=port->state;
/*
*Thismeansyoucalledthisfunction_after_theportwas
*closed.Nocookieforyou.
*/
BUG_ON(!state);
/*函数唤醒与state->port相关联的终端。*/
tty_wakeup(state->port.tty);
}

uart_suspend_port

uart_suspend_port函数用于将端口挂起以进行电源管理。它执行一系列操作,包括检查子设备是否可以唤醒系统,停止发送和接收数据,等待发送缓冲区为空,关闭端口,停止控制台,并更改端口的电源管理状态。

intuart_suspend_port(structuart_driver*drv,structuart_port*uport)
{
structuart_state*state=drv->state+uport->line;
structtty_port*port=&state->port;
structdevice*tty_dev;
structuart_matchmatch={uport,drv};

/*给port加锁,以确保在执行其他操作时不会发生竞争条件*/
mutex_lock(&port->mutex);

/*查找与uport->dev相关联的子设备。它使用match结构体和serial_match_port函数来匹配子设备*/
tty_dev=device_find_child(uport->dev,&match,serial_match_port);
/*如果找到了子设备并且该设备可以唤醒系统,则将uport->irq设置为唤醒中断,并将uport->irq_wake设置为1。然后,释放tty_dev并解锁port的互斥锁,并返回0*/
if(device_may_wakeup(tty_dev)){
if(!enable_irq_wake(uport->irq))
uport->irq_wake=1;
put_device(tty_dev);
mutex_unlock(&port->mutex);
return0;
}
/*如果找到了子设备但该设备不能唤醒系统,则释放tty_dev*/
put_device(tty_dev);

/*Nothingtodoiftheconsoleisnotsuspending*/
/*如果控制台未启用挂起并且uport是控制台,则跳转到unlock解锁*/
if(!console_suspend_enabled&&uart_console(uport))
gotounlock;

/*将uport->suspended设置为1,表示端口已挂起。*/
uport->suspended=1;

/*如果端口已初始化,则执行一些操作以停止传输并关闭端口。这些操作包括设置ASYNCB_SUSPENDED和清除ASYNCB_INITIALIZED标志,停止发送和接收数据,等待发送缓冲区为空,关闭端口*/
if(port->flags&ASYNC_INITIALIZED){
conststructuart_ops*ops=uport->ops;
inttries;

set_bit(ASYNCB_SUSPENDED,&port->flags);
clear_bit(ASYNCB_INITIALIZED,&port->flags);

spin_lock_irq(&uport->lock);
ops->stop_tx(uport);
ops->set_mctrl(uport,0);
ops->stop_rx(uport);
spin_unlock_irq(&uport->lock);

/*
*Waitforthetransmittertoempty.
*/
for(tries=3;!ops->tx_empty(uport)&&tries;tries--)
msleep(10);
if(!tries)
dev_err(uport->dev,"%s%d:Unabletodraintransmitter
",
drv->dev_name,
drv->tty_driver->name_base+uport->line);

ops->shutdown(uport);
}

/*
*Disabletheconsoledevicebeforesuspending.
*/
/**/
/*如果uport是控制台,则停止控制台*/
if(uart_console(uport))
console_stop(uport->cons);
/*调用uart_change_pm函数以更改端口的电源管理状态为UART_PM_STATE_OFF*/
uart_change_pm(state,UART_PM_STATE_OFF);
unlock:
mutex_unlock(&port->mutex);

return0;
}

uart_resume_port

uart_resume_port作用是恢复一个已经挂起的UART端口。

intuart_resume_port(structuart_driver*drv,structuart_port*uport)
{
structuart_state*state=drv->state+uport->line;
structtty_port*port=&state->port;
structdevice*tty_dev;
structuart_matchmatch={uport,drv};
structktermiostermios;

mutex_lock(&port->mutex);

/*使用device_find_child搜索与名为match的structuart_match匹配的uport->dev的子设备*/
tty_dev=device_find_child(uport->dev,&match,serial_match_port);

/*如果找到设备并且端口未挂起并且设备可以唤醒,则函数禁用IRQ唤醒并返回0*/
if(!uport->suspended&&device_may_wakeup(tty_dev)){
if(uport->irq_wake){
disable_irq_wake(uport->irq);
uport->irq_wake=0;
}
put_device(tty_dev);
mutex_unlock(&port->mutex);
return0;
}

/*函数将uport->suspended设置为0*/
put_device(tty_dev);
uport->suspended=0;

/*
*Re-enabletheconsoledeviceaftersuspending.
*/
/*如果端口是控制台端口,则函数将termios结构设置为控制台cflag设置*/
if(uart_console(uport)){
/*
*Firsttrytousetheconsolecflagsetting.
*/
memset(&termios,0,sizeof(structktermios));
termios.c_cflag=uport->cons->cflag;

/*
*Ifthat'sunset,usethettytermiossetting.
*/
if(port->tty&&termios.c_cflag==0)
termios=port->tty->termios;
/*如果启用了控制台挂起,则函数使用uart_change_pm将电源管理状态更改为打开状态,使用uport->ops->set_termios设置termios,并使用console_start启动控制台*/
if(console_suspend_enabled)
uart_change_pm(state,UART_PM_STATE_ON);
uport->ops->set_termios(uport,&termios,NULL);
if(console_suspend_enabled)
console_start(uport->cons);
}

if(port->flags&ASYNC_SUSPENDED){
conststructuart_ops*ops=uport->ops;
intret;
/*如果端口已挂起,则函数使用uart_change_pm将电源管理状态更改为打开状态*/
uart_change_pm(state,UART_PM_STATE_ON);
spin_lock_irq(&uport->lock);
/*使用ops->set_mctrl将调制解调器控制线设置为0*/
ops->set_mctrl(uport,0);
spin_unlock_irq(&uport->lock);
if(console_suspend_enabled||!uart_console(uport)){
/*Protectedbyportmutexfornow*/
structtty_struct*tty=port->tty;
/*使用ops->startup启动端口*/
ret=ops->startup(uport);
if(ret==0){
/*如果端口成功启动,则使用uart_change_speed更改端口速度,使用ops->start_tx启动传输,并在port->flags中设置ASYNCB_INITIALIZED位*/
if(tty)
uart_change_speed(tty,state,NULL);
spin_lock_irq(&uport->lock);
ops->set_mctrl(uport,uport->mctrl);

ops->start_tx(uport);
spin_unlock_irq(&uport->lock);
set_bit(ASYNCB_INITIALIZED,&port->flags);
}else{
/*
*Failedtoresume-maybehardwarewentaway?
*Clearthe"initialized"flagsowewon'ttry
*tocallthelowleveldriversshutdownmethod.
*/
/*如果端口无法恢复,则函数清除ASYNCB_INITIALIZED位并调用uart_shutdown*/
uart_shutdown(tty,state);
}
}

clear_bit(ASYNCB_SUSPENDED,&port->flags);
}

mutex_unlock(&port->mutex);

return0;
}

uart_get_baud_rate

uart_get_baud_rate,该函数的作用是根据给定的终端设置和范围,获取一个可用的波特率。如果无法获取满足要求的波特率,则会尽可能地使用最接近的波特率。

/**
*uart_get_baud_rate-returnbaudrateforaparticularport
*@port:uart_portstructuredescribingtheportinquestion.
*@termios:desiredtermiossettings.
*@old:oldtermios(orNULL)
*@min:minimumacceptablebaudrate
*@max:maximumacceptablebaudrate
*
*Decodethetermiosstructureintoanumericbaudrate,
*takingaccountofthemagic38400baudrate(withspd_*
*flags),andmappingthe%B0rateto9600baud.
*
*Ifthenewbaudrateisinvalid,trytheoldtermiossetting.
*Ifit'sstillinvalid,wetry9600baud.
*
*Updatethe@termiosstructuretoreflectthebaudrate
*we'reactuallygoingtobeusing.Don'tdothisforthecase
*whereB0isrequested("hangup").
*/
unsignedint
uart_get_baud_rate(structuart_port*port,structktermios*termios,
structktermios*old,unsignedintmin,unsignedintmax)
{
unsignedinttry;
unsignedintbaud;
unsignedintaltbaud;
inthung_up=0;
upf_tflags=port->flags&UPF_SPD_MASK;

switch(flags){
caseUPF_SPD_HI:
altbaud=57600;
break;
caseUPF_SPD_VHI:
altbaud=115200;
break;
caseUPF_SPD_SHI:
altbaud=230400;
break;
caseUPF_SPD_WARP:
altbaud=460800;
break;
default:
altbaud=38400;
break;
}

for(try=0;try< 2; try++) {
  baud = tty_termios_baud_rate(termios);

  /*
   * The spd_hi, spd_vhi, spd_shi, spd_warp kludge...
   * Die! Die! Die!
   */
  if (try == 0 && baud == 38400)
   baud = altbaud;

  /*
   * Special case: B0 rate.
   */
  if (baud == 0) {
   hung_up = 1;
   baud = 9600;
  }

  if (baud >=min&&baud<= max)
   return baud;

  /*
   * Oops, the quotient was zero.  Try again with
   * the old baud rate if possible.
   */
  termios->c_cflag&=~CBAUD;
if(old){
baud=tty_termios_baud_rate(old);
if(!hung_up)
tty_termios_encode_baud_rate(termios,
baud,baud);
old=NULL;
continue;
}

/*
*Asalastresort,iftherangecannotbemetthenclipto
*thenearestchipsupportedrate.
*/
if(!hung_up){
if(baud<= min)
    tty_termios_encode_baud_rate(termios,
       min + 1, min + 1);
   else
    tty_termios_encode_baud_rate(termios,
       max - 1, max - 1);
  }
 }
 /* Should never happen */
 WARN_ON(1);
 return 0;
}

该函数所作工作如下

根据 UPF_SPD_MASK 标志位解析出一个备用波特率 altbaud。

函数会尝试两次获取波特率。第一次,函数会从 termios 中解析出当前波特率,如果它等于 38400,则将波特率设置为备用波特率 altbaud。如果波特率等于 0(即请求“挂起”),则设置波特率为 9600。如果波特率在 min 和 max 范围内,则返回该波特率。

如果第一次获取的波特率为 0,则函数会尝试使用旧的终端设置。

如果仍然无法满足要求,函数会将波特率剪裁到最接近的支持的波特率。剪裁的方式是,如果波特率小于等于最小值 min,则设置波特率为 min + 1,否则设置波特率为 max - 1。

uart_get_divisor

uart_get_divisor,用于计算给定端口的 UART 时钟分频器值,以实现指定的波特率 。主要是通过一些基本的数学运算来计算出 UART 时钟分频器值。这个值是用来配置 UART 硬件的,以实现指定的波特率。

在串口通信中,时钟分频器值对应着波特率,即时钟分频器值越小,波特率越高,传输速度越快。

/**
*uart_get_divisor-returnuartclockdivisor
*@port:uart_portstructuredescribingtheport.
*@baud:desiredbaudrate
*
*Calculatetheuartclockdivisorfortheport.
*/
unsignedint
uart_get_divisor(structuart_port*port,unsignedintbaud)
{
unsignedintquot;

/*
*Oldcustomspeedhandling.
*/
if(baud==38400&&(port->flags&UPF_SPD_MASK)==UPF_SPD_CUST)
quot=port->custom_divisor;
else
quot=DIV_ROUND_CLOSEST(port->uartclk,16*baud);

returnquot;
}

该函数所作工作如下

首先根据给定的波特率计算出 UART 时钟周期的长度 period,这个周期的长度是通过 16 * baud 计算得到的。然后,将端口的 UART 时钟频率除以 period,得到的商值 quot 就是需要的 UART 时钟分频器值。这里使用了 DIV_ROUND_CLOSEST 宏,它的作用是将浮点数四舍五入为最接近的整数值。

在计算时钟分频器值时,还有一个特殊的情况需要处理。如果给定的波特率为 38400,并且端口的标志位值为 UPF_SPD_CUST,则需要使用端口的自定义分频器值,而不是根据公式计算出来的值。这是因为在一些老的串口驱动中,可能会使用自定义分频器值来支持一些特殊的波特率。

uart_update_timeout

uart_update_timeout用于设置串口的 FIFO 超时时间。FIFO(First-In-First-Out)是串口硬件中用于缓存数据的一种常见结构,它可以提高串口传输的效率。而超时时间则是指在 FIFO 中没有数据传输时,等待多长时间后自动清空 FIFO。超时时间的设置可以影响串口传输的稳定性和效率。

/**
*uart_update_timeout-updateper-portFIFOtimeout.
*@port:uart_portstructuredescribingtheport
*@cflag:termioscflagvalue
*@baud:speedoftheport
*
*SettheportFIFOtimeoutvalue.The@cflagvalueshould
*reflecttheactualhardwaresettings.
*/
void
uart_update_timeout(structuart_port*port,unsignedintcflag,
unsignedintbaud)
{
unsignedintbits;

/*bytesizeandparity*/
switch(cflag&CSIZE){
caseCS5:
bits=7;
break;
caseCS6:
bits=8;
break;
caseCS7:
bits=9;
break;
default:
bits=10;
break;/*CS8*/
}

if(cflag&CSTOPB)
bits++;
if(cflag&PARENB)
bits++;

/*
*Thetotalnumberofbitstobetransmittedinthefifo.
*/
bits=bits*port->fifosize;

/*
*Figurethetimeouttosendtheabovenumberofbits.
*Add.02secondsofslop
*/
port->timeout=(HZ*bits)/baud+HZ/50;
}

根据终端设置中的 cflag 值,计算出每个字节需要传输的位数 bits。根据 cflag 中的 CSIZE 标志位,确定每个字节的位数(5、6、7 或 8 位),并根据 CSTOPB 和 PARENB 标志位,增加停止位和奇偶校验位的位数。

将每个字节需要传输的位数 bits 乘以 FIFO 的大小,得到总共需要传输的位数。

根据波特率和总共需要传输的位数,计算出超时时间。将总共需要传输的位数除以波特率,得到传输这些数据所需要的时间,再加上一些额外的时间(0.02 秒)作为缓冲,得到超时时间。

最后,将计算出来的超时时间赋值给端口结构体中的 timeout 成员变量,从而完成 FIFO 超时时间的设置。

uart_match_port

uart_match_port根据两个端口的属性比较两个串口端口是否相等。

/*
*Arethetwoportsequivalent?
*/
intuart_match_port(structuart_port*port1,structuart_port*port2)
{
if(port1->iotype!=port2->iotype)
return0;

switch(port1->iotype){
caseUPIO_PORT:
return(port1->iobase==port2->iobase);
caseUPIO_HUB6:
return(port1->iobase==port2->iobase)&&
(port1->hub6==port2->hub6);
caseUPIO_MEM:
caseUPIO_MEM32:
caseUPIO_MEM32BE:
caseUPIO_AU:
caseUPIO_TSI:
return(port1->mapbase==port2->mapbase);
}
return0;
}

根据两个串口端口的 iotype 属性进行比较,如果不相等,则两个端口不相等,函数返回 0。

根据 iotype 属性的不同,比较两个端口的其他属性。对于 UPIO_PORT 和 UPIO_HUB6 类型的端口,比较它们的 iobase 和 hub6 属性是否相等;对于其他类型的端口,比较它们的 mapbase 属性是否相等。如果所有属性都相等,则两个端口相等,函数返回 1,否则返回 0。

uart_console_write

uart_console_write用于将控制台消息写入串口。

嵌入式系统中,通常需要将控制台输出重定向到串口,以便进行调试和日志记录。该函数实现了将一个字符串写入串口的操作,其中需要将字符串中的换行符转换为回车换行符。

/**
*uart_console_write-writeaconsolemessagetoaserialport
*@port:theporttowritethemessage
*@s:arrayofcharacters
*@count:numberofcharactersinstringtowrite
*@putchar:functiontowritecharactertoport
*/
voiduart_console_write(structuart_port*port,constchar*s,
unsignedintcount,
void(*putchar)(structuart_port*,int))
{
unsignedinti;

for(i=0;i< count; i++, s++) {
  if (*s == '
')
   putchar(port, '
');
  putchar(port, *s);
 }
}

该函数的实现主要是遍历字符串中的所有字符,并将每个字符写入串口。在写入字符之前,需要判断该字符是否为换行符。如果是换行符,则需要先将其转换为回车换行符,再写入串口。

总结

对接底层的部分,Kernel 主要是提供了两个接口:

1、uart_register_driver (一次调用)

2、uart_add_one_port (多次调用)

通过这两个接口,实现了芯片将自己的 UART 对接到 Linux Kernel UART Driver 中。

芯片厂商需要自行设计并实现的部分有:

1、uart_drvier 结构(一个)

2、uart_port 结构(多个)

3、uart_ops 对串口的操作集(可能一个,可能多个)

所以从结构上来看,整个对接过程为:

71ea8d618cd911e8ca713408e39c1bbd.png
b1f23b238e2e3853a9dab36cc3dcd8ca.png

这里有一点需要特别注意,在对接底层的部分中,Kernel 定义了一个结构体叫:struct uart_ops

在 tty 层,对 tty_driver 初始化的时候(serial_core.c),调用到:

tty_set_operations(normal,&uart_ops);

而他的实现是:

voidtty_set_operations(structtty_driver*driver,conststructtty_operations*op)

{
driver->ops=op;
};
EXPORT_SYMBOL(tty_set_operations);

看到了么,传进去的是 ****tty_operations *op****,所以,在 tty_driver 挂接的 uart_ops 并非那个 struct uart_ops,而是这个 serial_core.c 文件内定义的:

staticconststructtty_operationsuart_ops={
.open=uart_open,
.close=uart_close,
.write=uart_write,
.put_char=uart_put_char,
.flush_chars=uart_flush_chars,
.write_room=uart_write_room,
.chars_in_buffer=uart_chars_in_buffer,
.flush_buffer=uart_flush_buffer,
.ioctl=uart_ioctl,
.throttle=uart_throttle,
.unthrottle=uart_unthrottle,
.send_xchar=uart_send_xchar,
.set_termios=uart_set_termios,
.set_ldisc=uart_set_ldisc,
.stop=uart_stop,
.start=uart_start,
.hangup=uart_hangup,
.break_ctl=uart_break_ctl,
.wait_until_sent=uart_wait_until_sent,
#ifdefCONFIG_PROC_FS
.proc_show=uart_proc_show,
#endif
.tiocmget=uart_tiocmget,
.tiocmset=uart_tiocmset,
.set_serial=uart_set_info_user,
.get_serial=uart_get_info_user,
.get_icount=uart_get_icount,
#ifdefCONFIG_CONSOLE_POLL
.poll_init=uart_poll_init,
.poll_get_char=uart_poll_get_char,
.poll_put_char=uart_poll_put_char,
#endif
};

名字一样,但是不是同一个结构,容易让人眼花~~

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • uart
    +关注

    关注

    22

    文章

    1235

    浏览量

    101334
  • 数据结构
    +关注

    关注

    3

    文章

    573

    浏览量

    40118
  • 串口驱动
    +关注

    关注

    2

    文章

    82

    浏览量

    18644

原文标题:【驱动】串口驱动分析(三)-serial driver

文章出处:【微信号:嵌入式与Linux那些事,微信公众号:嵌入式与Linux那些事】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    【开源的串口可视化工具——Serial Studio】

    分享一个开源的串口项目——Serial Studio,这是一个强大的数据可视化软件,支持串口通信,串口终端,网络通信 TCP/UDP,MQTT通信协议。这个项目遵循MIT协议,所以是可
    的头像 发表于 01-18 15:03 1.2w次阅读
    【开源的<b class='flag-5'>串口</b>可视化工具——<b class='flag-5'>Serial</b> Studio】

    串口设备框架serial_v2源码分析-阻塞模式

    serial_v2中,串口设备以应用层视角,即阻塞模式或非阻塞模式来作为该串口设备的开启标志.
    的头像 发表于 09-14 11:34 2058次阅读

    USB To Serial Driver

    USB To Serial Driver  
    发表于 09-28 15:57

    关于串口调试助手和virtual serial port driver

    想问下proteus做单片机和pc通信提示需要virtual serial port driver串口调试助手,想问下怎么用,下载下来的这两个好像不起作用
    发表于 11-17 16:55

    请问driver/tty/serial/8250/8250-omap.c驱动是什么串口驱动

    是CONFIG_SERIAL_8250,而不是CONFIG_SERIAL_OMAP,那么这里为什么要配置成8250呢?8250的串口驱动是干嘛用的?
    发表于 06-21 02:01

    什么是串口驱动器?

    什么是串口驱动器?串口驱动的作用?以及它如何适用于GSM M66模块。以上来自于谷歌翻译以下为原文 what is serial
    发表于 01-14 15:24

    虚拟串口Virtual Serial Port Driver使用报错怎么解决?

    使用虚拟串口到底是干什么?虚拟串口Virtual Serial Port Driver使用报错怎么解决?
    发表于 02-22 06:57

    Open Universal Serial Bus Driv

    Open Universal Serial Bus Driver Interface (OpenUSBDI) Specification This document specifies
    发表于 04-11 19:16 15次下载

    Serial Monitor (串口监视、检测、分析工具)v

    Serial Monitor:Serial Monitor是一款功能强大的串口监视、检测、分析工具,软件使用更加简单,尤其适合开发人员使用。 
    发表于 05-26 09:03 86次下载

    Proteus串口资料COMPIM Serial Port

    Proteus串口资料COMPIM Serial Port Model The COMPIM model is a Physical Interface Model of a serial port. Incoming
    发表于 04-17 16:13 0次下载

    Virtual Serial Port Driver 6.9(虚拟串口)

    电子发烧友网站提供《Virtual Serial Port Driver 6.9(虚拟串口).rar》资料免费下载
    发表于 08-02 00:00 39次下载

    经典实用USB转串口驱动STM32 Virtual COM Port Driver(V1.3.1)

    电子发烧友网站提供《经典实用USB转串口驱动STM32 Virtual COM Port Driver(V1.3.1).zip》资料免费下载
    发表于 08-01 10:32 1289次下载

    matlab中的串口软件serial_1

    电子发烧友网站提供《matlab中的串口软件serial_1.zip》资料免费下载
    发表于 07-07 16:32 1次下载

    PL-2303 Vista Driver Installer12

    PL-2303 Vista Driver Installer串口驱动
    发表于 12-09 15:45 1次下载

    NodeMCU V3.0 Arduino开发串口使用

    NodeMCU V3.0 Arduino开发串口使用串口使用串口使用void setup() { // put your setup code here, to run once
    发表于 11-16 09:51 1次下载
    NodeMCU V3.0 Arduino开发<b class='flag-5'>之</b><b class='flag-5'>串口</b>使用