技術(shù)文章:基于Linux的tty架構(gòu)及UART驅(qū)動(dòng)詳解
2.2.3. struct uart_state
每一個(gè)uart端口對(duì)應(yīng)著一個(gè)uart_state,該結(jié)構(gòu)體將uart_port與對(duì)應(yīng)的circ_buf聯(lián)系起來(lái)。uart_state有兩個(gè)成員在底層串口驅(qū)動(dòng)會(huì)用到:xmit和port。
用戶空間程序通過(guò)串口發(fā)送數(shù)據(jù)時(shí),上層驅(qū)動(dòng)將用戶數(shù)據(jù)保存在xmit;而串口發(fā)送中斷處理函數(shù)就是通過(guò)xmit獲取到用戶數(shù)據(jù)并將它們發(fā)送出去。串口接收中斷處理函數(shù)需要通過(guò)port將接收到的數(shù)據(jù)傳遞給線路規(guī)程層。
struct uart_state {
struct tty_port port;
enum uart_pm_state pm_state;
struct circ_buf xmit;
struct uart_port *uart_port; 對(duì)應(yīng)于一個(gè)串口設(shè)備
};
2.2.4. struct uart_port
uart_port用于描述串口端口的I/O端口或I/O內(nèi)存地址、FIFO大小、端口類型、串口時(shí)鐘等信息。實(shí)際上,一個(gè)uart_port實(shí)現(xiàn)對(duì)應(yīng)一個(gè)串口設(shè)備。
struct uart_port {
spinlock_t lock; port lock
unsigned long iobase; in/out[bwl]
unsigned char __iomem *membase; read/write[bwl]
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);
int (*handle_irq)(struct uart_port *);
void (*pm)(struct uart_port *, unsigned int state,
unsigned int old);
void (*handle_break)(struct uart_port *);
unsigned int irq; irq number
unsigned long irqflags; irq flags
unsigned int uartclk; base uart clock
unsigned int fifosize; tx fifo size
unsigned char x_char; xon/xoff char
unsigned char regshift; reg offset shift
unsigned char iotype; io access style
unsigned char unused1;
#define UPIO_PORT (0)
#define UPIO_HUB6 (1)
#define UPIO_M(jìn)EM (2)
#define UPIO_M(jìn)EM32 (3)
#define UPIO_AU (4) Au1x00 and RT288x type IO
#define UPIO_TSI (5) Tsi108/109 type IO
unsigned int read_status_mask; driver specific
unsigned int ignore_status_mask; driver specific
struct uart_state *state; pointer to parent state
struct uart_icount icount; statistics
struct console *cons; struct console, if any
#if defined(CONFIG_SERIAL_CORE_CONSOLE) || defined(SUPPORT_SYSRQ)
unsigned long sysrq; sysrq timeout
#endif
upf_t flags;
#define UPF_FOURPORT ((__force upf_t) (1 << 1))
#define UPF_SAK ((__force upf_t) (1 << 2))
#define UPF_SPD_M(jìn)ASK ((__force upf_t) (0x1030))
#define UPF_SPD_HI ((__force upf_t) (0x0010))
#define UPF_SPD_VHI ((__force upf_t) (0x0020))
#define UPF_SPD_CUST ((__force upf_t) (0x0030))
#define UPF_SPD_SHI ((__force upf_t) (0x1000))
#define UPF_SPD_WARP ((__force upf_t) (0x1010))
#define UPF_SKIP_TEST ((__force upf_t) (1 << 6))
#define UPF_AUTO_IRQ ((__force upf_t) (1 << 7))
#define UPF_HARDPPS_CD ((__force upf_t) (1 << 11))
#define UPF_LOW_LATENCY ((__force upf_t) (1 << 13))
#define UPF_BUGGY_UART ((__force upf_t) (1 << 14))
#define UPF_NO_TXEN_TEST ((__force upf_t) (1 << 15))
#define UPF_M(jìn)AGIC_M(jìn)ULTIPLIER ((__force upf_t) (1 << 16))
Port has hardware-assisted h/w flow control (iow, auto-RTS *not* auto-CTS)
#define UPF_HARD_FLOW ((__force upf_t) (1 << 21))
Port has hardware-assisted s/w flow control
#define UPF_SOFT_FLOW ((__force upf_t) (1 << 22))
#define UPF_CONS_FLOW ((__force upf_t) (1 << 23))
#define UPF_SHARE_IRQ ((__force upf_t) (1 << 24))
#define UPF_EXAR_EFR ((__force upf_t) (1 << 25))
#define UPF_BUG_THRE ((__force upf_t) (1 << 26))
The exact UART type is known and should not be probed.
#define UPF_FIXED_TYPE ((__force upf_t) (1 << 27))
#define UPF_BOOT_AUTOCONF ((__force upf_t) (1 << 28))
#define UPF_FIXED_PORT ((__force upf_t) (1 << 29))
#define UPF_DEAD ((__force upf_t) (1 << 30))
#define UPF_IOREMAP ((__force upf_t) (1 << 31))
#define UPF_CHANGE_M(jìn)ASK ((__force upf_t) (0x17fff))
#define UPF_USR_M(jìn)ASK ((__force upf_t) (UPF_SPD_M(jìn)ASK|UPF_LOW_LATENCY))
unsigned int mctrl; current modem ctrl settings
unsigned int timeout; character-based timeout
unsigned int type; port type
const struct uart_ops *ops;
unsigned int custom_divisor;
unsigned int line; port index
resource_size_t mapbase; for ioremap
struct device *dev; parent device
unsigned char hub6; this should be in the 8250 driver
unsigned char suspended;
unsigned char irq_wake;
unsigned char unused[2];
void *private_data; generic platform data pointer
};
2.2.5. struct uart_ops
struct uart_ops涵蓋了驅(qū)動(dòng)可對(duì)串口的所有操作
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 *);
void (*shutdown)(struct uart_port *);
void (*flush_buffer)(struct uart_port *);
void (*set_termios)(struct uart_port *, struct ktermios *new,
struct ktermios *old);
void (*set_ldisc)(struct uart_port *, int new);
void (*pm)(struct uart_port *, unsigned int state,
unsigned int oldstate);
int (*set_wake)(struct uart_port *, unsigned int state);
* 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 *);
* Request IO and memory resources used by the port.
* This includes iomapping the port if necessary.
int (*request_port)(struct uart_port *);
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);
#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
};
2.3. 關(guān)鍵流程2.3.1. 注冊(cè)流程2.3.1.1. 注冊(cè)u(píng)art_driver
此接口在uart driver中調(diào)用,用來(lái)注冊(cè)u(píng)art_driver到kernel中,調(diào)用階段在uart driver的初始階段,例如:module_init(), uart_driver的注冊(cè)流程圖
圖3.3uart driver注冊(cè)流程
注冊(cè)過(guò)程主要做了以下操作:
1、根據(jù)driver支持的最大設(shè)備數(shù),申請(qǐng)n個(gè)uart_state空間,每一個(gè)uart_state都有一個(gè)uart_port。2、分配一個(gè)tty_driver,并將uart_driver->tty_driver指向它。3、對(duì)tty_driver進(jìn)行設(shè)置,其中包括默認(rèn)波特率、檢驗(yàn)方式等,還有一個(gè)重要的ops,結(jié)構(gòu)體tty_operation的注冊(cè),它是tty核心與串口驅(qū)動(dòng)通信的接口。4、初始化每一個(gè)uart_state的tty_port;5、注冊(cè)tty_driver。注冊(cè)u(píng)art_driver實(shí)際上是注冊(cè)tty_driver,與用戶空間打交道的工作完全交給tty_driver,這一部分是內(nèi)核實(shí)現(xiàn)好的不需要修改2.3.1.2. 添加uart_port
此接口用于注冊(cè)一個(gè)uart port 到uart driver上,通過(guò)注冊(cè),uart driver就可以訪問(wèn)對(duì)應(yīng)的uart port,進(jìn)行數(shù)據(jù)收發(fā)。該接口在uart driver中的probe函數(shù)調(diào)用,必須保證晚于uart_register_drver的注冊(cè)過(guò)程。

發(fā)表評(píng)論
請(qǐng)輸入評(píng)論內(nèi)容...
請(qǐng)輸入評(píng)論/評(píng)論長(zhǎng)度6~500個(gè)字
圖片新聞
-
馬云重返一線督戰(zhàn),阿里重啟創(chuàng)始人模式
-
機(jī)器人奧運(yùn)會(huì)戰(zhàn)報(bào):宇樹(shù)機(jī)器人摘下首金,天工Ultra搶走首位“百米飛人”
-
存儲(chǔ)圈掐架!江波龍起訴佰維,索賠121萬(wàn)
-
長(zhǎng)安汽車母公司突然更名:從“中國(guó)長(zhǎng)安”到“辰致科技”
-
豆包前負(fù)責(zé)人喬木出軌BP后續(xù):均被辭退
-
字節(jié)AI Lab負(fù)責(zé)人李航卸任后返聘,Seed進(jìn)入調(diào)整期
-
員工持股爆雷?廣汽埃安緊急回應(yīng)
-
中國(guó)“智造”背后的「關(guān)鍵力量」
最新活動(dòng)更多
-
10月23日火熱報(bào)名中>> 2025是德科技創(chuàng)新技術(shù)峰會(huì)
-
10月23日立即報(bào)名>> Works With 開(kāi)發(fā)者大會(huì)深圳站
-
10月24日立即參評(píng)>> 【評(píng)選】維科杯·OFweek 2025(第十屆)物聯(lián)網(wǎng)行業(yè)年度評(píng)選
-
即日-11.25立即下載>>> 費(fèi)斯托白皮書(shū)《柔性:汽車生產(chǎn)未來(lái)的關(guān)鍵》
-
11月27日立即報(bào)名>> 【工程師系列】汽車電子技術(shù)在線大會(huì)
-
12月18日立即報(bào)名>> 【線下會(huì)議】OFweek 2025(第十屆)物聯(lián)網(wǎng)產(chǎn)業(yè)大會(huì)
推薦專題
- 1 特斯拉工人被故障機(jī)器人打成重傷,索賠3.6億
- 2 【行業(yè)深度研究】退居幕后四年后,張一鳴終于把算法公司變成AI公司?
- 3 人形機(jī)器人廠商,正在批量復(fù)刻宇樹(shù)G1
- 4 AI 時(shí)代,阿里云想當(dāng)“安卓” ,那誰(shuí)是“蘋(píng)果”?
- 5 華為公布昇騰芯片三年計(jì)劃,自研HBM曝光
- 6 硬剛英偉達(dá)!華為發(fā)布全球最強(qiáng)算力超節(jié)點(diǎn)和集群
- 7 機(jī)器人9月大事件|3家國(guó)產(chǎn)機(jī)器人沖刺IPO,行業(yè)交付與融資再創(chuàng)新高!
- 8 谷歌“香蕉”爆火啟示:國(guó)產(chǎn)垂類AI的危機(jī)還是轉(zhuǎn)機(jī)?
- 9 00后華裔女生靠?jī)刹緼I電影狂賺7.8億人民幣,AI正式進(jìn)軍好萊塢
- 10 美光:AI Capex瘋投不止,終于要拉起存儲(chǔ)超級(jí)周期了?