1. 硬件體驗
使用 Linux 自帶的 USB Gadget 驅動 /drivers/usb/gadget/legacy/serial.c
使用 USB 線,連接板子的 OTG 口和 PC 的 USB 口。
然后在板子加載驅動程序后,可以看到新的設備節點 /dev/ttyGS0:
# modprobe g_serial
g_serial gadget: Gadget Serial v2.4
g_serial gadget: g_serial ready
g_serial gadget: high-speed config #2: CDC ACM config
# ls /dev/ttyGS0 -l
crw-rw---- 1 root dialout 246, 0 Jan 1 00:30 /dev/ttyGS0
在 PC 上,如果是 Windows 系統,可以在設備管理器里看到新的 USB 串口:
在 PC 上,如果是 VMware 上的 Linux 系統,按下圖操作,先把 USB 串口連接到 VMware:
然后在 PC Linux 中可以看到新的設備節點:
book@100ask:~$ dmesg
[ 286.903239] usb 1-1: new high-speed USB device number 2 using ehci-pci
[ 287.254549] usb 1-1: New USB device found, idVendor=0525, idProduct=a4a7, bcdDevice= 4.09
[ 287.254550] usb 1-1: New USB device strings: Mfr=1, Product=2, SerialNumber=0
[ 287.254551] usb 1-1: Product: Gadget Serial v2.4
[ 287.254552] usb 1-1: Manufacturer: Linux 4.9.88 with 2184000.usb
[ 287.342786] cdc_acm 1-1:2.0: ttyACM0: USB ACM device
[ 287.343202] usbcore: registered new interface driver cdc_acm
[ 287.343202] cdc_acm: USB Abstract Control Model driver for USB modems and ISDN adapters
book@100ask:~$ ls /dev/ttyACM0 -l
crw-rw---- 1 root dialout 166, 0 Mar 5 22:38 /dev/ttyACM0
2. Serial分析
2.1 軟件框架
Gadget 串口的框架如下:
u_serial 提供了有 2 種方法來使用 Gadget 串口:
- u_serial.c 里注冊 tty_driver 結構體 gs_tty_driver,在板子上編寫 APP 訪問設備 /dev/ttyGS0 即可與 Host 交互(Host 要打開 USB 串口)
- u_serial.c 里注冊 console 結構體 gserial_cons。啟動 Linux 內核時傳入 commandline 參數"console=ttyGS0"后,內核的 printk 的信息通過 Gadget 串口打印出來(Host 要打開 USB 串口):
注冊 TTY 和 console 的過程:
gs_bind // driversusbgadgetlegacyserial.c
status = serial_register_ports(cdev, &serial_config_driver,"acm");
fi_serial[i] = usb_get_function_instance(f_name);
acm_alloc_instance // driversusbgadgetfunctionf_acm.c
ret = gserial_alloc_line(&opts- >port_num); // driversusbgadgetfunctionu_serial.c
// 注冊TTY
tty_dev = tty_port_register_device(&ports[port_num].port- >port,
gs_tty_driver, port_num, NULL);
// 注冊console
gserial_console_init();
register_console(&gserial_cons);
2.2 數據傳輸
2.2.1 APP 訪問
注意,在 USB 中數據傳輸總是由 Host 發起,所以:
- 板子要事先準備好空間(設置好 out 方向的 usb_request 并放入隊列),以便接收 Host 發來的數據;
- 板子有數據想發送給 Host 時需要設置 in 方向的 usb_request,以便 Host 讀取。
板子上的 APP 訪問 /dev/ttyGS0 時,就會導致 gs_tty_ops 結構體的對應函數被調用:
APP 調用 open 函數時,會導致如下調用:
gs_open
gs_start_io(port);
// 取出 out 端點(對應 Host 來說是 out, 對于板子來說就是輸入)
struct usb_ep *ep = port- >port_usb- >out;
// 給 out 端點分配 usb_request
status = gs_alloc_requests(ep, head, gs_read_complete,&port- >read_allocated);
// 給 in 端點分配 usb_request, 但是在 open 時并沒有把 in 方向的 usb_request 放入隊列
status = gs_alloc_requests(port- >port_usb- >in, &port- >write_pool,gs_write_complete, &port- >write_allocated);
// 把 usb_request 放入隊列, 如果 Host 發來數據, 這個 usb_request 的 complete 函數被調用
started = gs_start_rx(port);
status = usb_ep_queue(out, req, GFP_ATOMIC);
APP 調用 write 函數時,會導致如下調用:
gs_write
gs_start_tx(port);
// 把 usb_request 放入隊列, Host讀取數據時就可以從中得到數據
status = usb_ep_queue(in, req, GFP_ATOMIC);
2.2.2 printk
啟動 Linux 內核時傳入 commandline 參數"console=ttyGS0"后,內核的 printk 的信息通過 Gadget 串口打印出來(Host 要打開 USB 串口)。
內核的 printk 函數會導致 gserial_cons 結構體中的 write 指針即gs_console_write
函數被調用:
gs_console_write 函數的調用關系如下:
gs_console_write
// 把要打印的數據放入環形 buffer
gs_buf_put(&info- >con_buf, buf, count);
// 喚醒內核線程
wake_up_process(info- >console_thread);
// 內核線程
gs_console_thread
// 被喚醒后
// 取出輸入端點和它的 usb_request
req = info- >console_req;
ep = port- >port_usb- >in;
// 從環形 buffer 得到數據、設置 usb_request
xfer = gs_buf_get(&info- >con_buf, req- >buf, size);
req- >length = xfer;
// 把 usb_request 放入隊列,以便 Host 讀取
ret = usb_ep_queue(ep, req, GFP_ATOMIC);
-
嵌入式
+關注
關注
5082文章
19104瀏覽量
304823 -
驅動
+關注
關注
12文章
1838瀏覽量
85262 -
Linux
+關注
關注
87文章
11292瀏覽量
209332
發布評論請先 登錄
相關推薦
評論