【随笔记】T507 Android10 EC200U-CN 4G Cat1 移植
基本信息
硬件信息
硬件平台:T507 (Android 10 Linux 4.9)
模组型号:EC200U-CN(Cat1)(展讯芯片)
相关文件
代理提供
longan/kernel/linux-4.9/drivers/net/usb/qmi_wwan_q.c
android/vendor/aw/public/prebuild/lib/librild/lib/lib32/libquectel-ril.so
android/vendor/aw/public/prebuild/lib/librild/lib/lib64/libquectel-ril.so
内核驱动
longan/kernel/linux-4.9/drivers/net/usb/Makefile
longan/kernel/linux-4.9/drivers/net/usb/qmi_wwan_q.c
longan/kernel/linux-4.9/drivers/usb/serial/option.c
longan/kernel/linux-4.9/drivers/usb/serial/usb_wwan.c
系统适配
android/hardware/ril/rild/rild.rc
android/hardware/ril/rild/radio.xml
android/device/softwinner/common/sepolicy/vendor/rild.te
android/vendor/aw/public/prebuild/lib/librild/radio_common.mk
android/vendor/aw/public/prebuild/lib/librild/lib/lib32/libquectel-ril.so
android/vendor/aw/public/prebuild/lib/librild/lib/lib64/libquectel-ril.so
驱动类型
-
GobiNet Driver
会生成 /dev/qcqmiX 设备节点和虚拟的 ethX 网卡(2.6.39 以下版本为 usbX)。
内核配置 驱动文件 CONFIG_USB_NET_DRIVERS
CONFIG_USB_USBNETlinux-4.9/drivers/net/usb/GobiNet/GobiUSBNet.c
linux-4.9/drivers/net/usb/GobiNet/QMIDevice.c
linux-4.9/drivers/net/usb/GobiNet/QMI.c -
qmi_wwan_q Driver
会生成 /dev/cdc-wdmX 设备节点和虚拟的 wwanX 网卡。
内核配置 驱动文件 CONFIG_USB_NET_DRIVERS
CONFIG_USB_USBNET
CONFIG_USB_NET_QMI_WWAN
CONFIG_USB_WDMlinux-4.9/drivers/net/usb/qmi_wwan.c
linux-4.9/drivers/net/usb/qmi_wwan_q.c -
ACM/ECM/RNDIS/NCM/MBIM Driver
会生成 /dev/ttyUSB0 ~ /dev/ttyUSB6 和虚拟的 usbX 网卡。
驱动类型 内核配置 驱动文件 ACM CONFIG_USB_ACM linux-4.9/drivers/usb/serial/option.c
linux-4.9/drivers/usb/serial/usb_wwan.c
linux-4.9/drivers/net/usb/cdc-acm.cECM CONFIG_USB_NET_DRIVERS
CONFIG_USB_USBNET
CONFIG_USB_NET_CDCETHERlinux-4.9/drivers/usb/serial/option.c
linux-4.9/drivers/usb/serial/usb_wwan.c
linux-4.9/drivers/net/usb/cdc-ether.cRNDIS CONFIG_USB_NET_DRIVERS
CONFIG_USB_USBNET
CONFIG_USB_NET_CDCETHERlinux-4.9/drivers/usb/serial/option.c
linux-4.9/drivers/usb/serial/usb_wwan.c
linux-4.9/drivers/net/usb/rndis_host.cNCM CONFIG_USB_NET_DRIVERS
CONFIG_USB_USBNET
CONFIG_USB_NET_CDCETHERlinux-4.9/drivers/usb/serial/option.c
linux-4.9/drivers/usb/serial/usb_wwan.c
linux-4.9/drivers/net/usb/cdc_ncm.cMBIM CONFIG_USB_NET_DRIVERS
CONFIG_USB_USBNET
CONFIG_USB_NET_CDCETHERlinux-4.9/drivers/usb/serial/option.c
linux-4.9/drivers/usb/serial/usb_wwan.c
linux-4.9/drivers/net/usb/cdc_mbim.c
驱动适配
这里使用 ECM 的驱动方式,参考《Quectel_UMTS_LTE_5G_Linux_USB_Driver_User_Guide_V3.0.pdf》
驱动移植
未涉及到电源控制,调试时使用的是开发板,手动给模组开机,因此下列会缺少电源控制的步骤。
-
涉及文件列表
// 需要修改的内核配置 longan/kernel/linux-4.9/.config // 需要修改的驱动文件 longan/kernel/linux-4.9/drivers/usb/serial/option.c longan/kernel/linux-4.9/drivers/usb/serial/usb_wwan.c // 需要用到的驱动文件 longan/kernel/linux-4.9/drivers/net/usb/cdc-acm.c longan/kernel/linux-4.9/drivers/net/usb/cdc-ether.c
-
在 option_ids 列表内增加 EC200U-CN 的 PID\VID, 这样才能识别到该 USB 设备为串口设备,文档内可查。
// linux-4.9/drivers/usb/serial/option.c static const struct usb_device_id option_ids[] = { #ifdef SUPPORT_QUECTEL { USB_DEVICE(0x2C7C, 0x0901) }, // support EC200U/EG915U #endif ...... }
-
一个 USB 设备可以有多个功能不同的接口,在 option_ids 添加该设备的 PID\VID 后,会导致该设备的所有接口都会绑定到 USB Serial Option 驱动上,导致 USBNet 驱动接口无法正常工作,因此需要在 option_probe 中根据类码、接口索引、端点数量、子类码将 USBNet 的接口排除出来。
// linux-4.9/drivers/usb/serial/option.c static int option_probe(struct usb_serial *serial, const struct usb_device_id *id) { ...... #ifdef SUPPORT_QUECTEL if (serial->dev->descriptor.idVendor == cpu_to_le16(0x2C7C)) { __u16 idProduct = le16_to_cpu(serial->dev->descriptor.idProduct); struct usb_interface_descriptor *intf = &serial->interface->cur_altsetting->desc; if (intf->bInterfaceClass != 0xFF || intf->bInterfaceSubClass == 0x42) { //ECM, RNDIS, NCM, MBIM, ACM, UAC, ADB return -ENODEV; } if ((idProduct&0xF000) == 0x0000) { //MDM interface 4 is QMI if (intf->bInterfaceNumber == 4 && intf->bNumEndpoints == 3 && intf->bInterfaceSubClass == 0xFF && intf->bInterfaceProtocol == 0xFF) return -ENODEV; } } #ifdef SUPPORT_QUECTEL_AUTO_SUSPEND //For USB Auto Suspend if (serial->dev->descriptor.idVendor == cpu_to_le16(0x2C7C)) { pm_runtime_set_autosuspend_delay(&serial->dev->dev, 3000); usb_enable_autosuspend(serial->dev); } #endif #ifdef SUPPORT_QUECTEL_REMOTE_WAKEUP //For USB Remote Wakeup if (serial->dev->descriptor.idVendor == cpu_to_le16(0x2C7C)) { device_init_wakeup(&serial->dev->dev, 1); //usb remote wakeup } #endif #endif /* Store the device flags so we can use them during attach. */ usb_set_serial_data(serial, (void *)device_flags); return 0; }
-
根据USB协议的要求,在批量输出传输期间,通过设置 URB_ZERO_PACKET 标志来添加处理零数据包的机制。
diff --git a/longan/kernel/linux-4.9/drivers/usb/serial/usb_wwan.c b/longan/kernel/linux-4.9/drivers/usb/serial/usb_wwan.c old mode 100644 new mode 100755 index 3dfdfc8..e56b275 --- a/longan/kernel/linux-4.9/drivers/usb/serial/usb_wwan.c +++ b/longan/kernel/linux-4.9/drivers/usb/serial/usb_wwan.c @@ -36,6 +36,8 @@ #include <linux/serial.h> #include "usb-wwan.h" +#define SUPPORT_QUECTEL 1 + /* * Generate DTR/RTS signals on the port using the SET_CONTROL_LINE_STATE request * in CDC ACM. @@ -504,6 +506,14 @@ static struct urb *usb_wwan_setup_urb(struct usb_serial_port *port, usb_fill_bulk_urb(urb, serial->dev, usb_sndbulkpipe(serial->dev, endpoint) | dir, buf, len, callback, ctx); + +#ifdef SUPPORT_QUECTEL + if (dir == USB_DIR_OUT) { + struct usb_device_descriptor *desc = &serial->dev->descriptor; + if (desc->idVendor == cpu_to_le16(0x2C7C)) + urb->transfer_flags |= URB_ZERO_PACKET; + } +#endif return urb; }
-
增加 USB 控制器复位后恢复操作
// linux-4.9/drivers/usb/serial/option.c static struct usb_serial_driver option_1port_device = { ...... #ifdef SUPPORT_QUECTEL .reset_resume = usb_wwan_resume, #endif };
-
在内核中启用 USB SERIAL 配置
CONFIG_USB_SERIAL=y CONFIG_USB_SERIAL_WWAN=y CONFIG_USB_SERIAL_OPTION=y CONFIG_USB_ACM=y CONFIG_USB_NET_DRIVERS=y CONFIG_USB_USBNET=y CONFIG_USB_NET_CDCETHER=y
完成标志
-
内核日志
[ 6.254850] usb 2-1: not running at top speed; connect to a high speed hub [ 6.284643] cdc_ether 2-1:1.0 usb0: register 'cdc_ether' at usb-sunxi-ohci-1, CDC Ethernet Device, 02:4b:b3:b9:eb:e5 [ 6.293812] vdc: Waited 0ms for vold [ 6.304202] option 2-1:1.2: GSM modem (1-port) converter detected [ 6.311813] usb 2-1: GSM modem (1-port) converter now attached to ttyUSB0 [ 6.321412] option 2-1:1.3: GSM modem (1-port) converter detected [ 6.328994] usb 2-1: GSM modem (1-port) converter now attached to ttyUSB1 [ 6.338217] option 2-1:1.4: GSM modem (1-port) converter detected [ 6.345935] usb 2-1: GSM modem (1-port) converter now attached to ttyUSB2 [ 6.355219] option 2-1:1.5: GSM modem (1-port) converter detected [ 6.362914] usb 2-1: GSM modem (1-port) converter now attached to ttyUSB3 [ 6.372287] option 2-1:1.6: GSM modem (1-port) converter detected [ 6.379951] usb 2-1: GSM modem (1-port) converter now attached to ttyUSB4 [ 6.389179] option 2-1:1.7: GSM modem (1-port) converter detected [ 6.396772] usb 2-1: GSM modem (1-port) converter now attached to ttyUSB5 [ 6.406217] option 2-1:1.8: GSM modem (1-port) converter detected [ 6.413736] usb 2-1: GSM modem (1-port) converter now attached to ttyUSB6
-
虚拟串口
mercury-demo:/ # ls -l /dev/ttyU* crwxrwxrwx 1 system system 188, 0 2022-12-30 21:00 /dev/ttyUSB0 crwxrwxrwx 1 system system 188, 1 2022-12-30 21:00 /dev/ttyUSB1 crwxrwxrwx 1 system system 188, 2 2022-12-30 21:00 /dev/ttyUSB2 crwxrwxrwx 1 system system 188, 3 2022-12-30 21:00 /dev/ttyUSB3 crw------- 1 root root 188, 4 2022-12-30 21:00 /dev/ttyUSB4 crw------- 1 root root 188, 5 2022-12-30 21:00 /dev/ttyUSB5 crw------- 1 root root 188, 6 2022-12-30 21:00 /dev/ttyUSB6
-
虚拟网卡
mercury-demo:/ # ifconfig -a ...... usb0 Link encap:Ethernet HWaddr 02:4b:b3:b9:eb:e5 Driver cdc_ether BROADCAST MULTICAST MTU:1500 Metric:1 RX packets:0 errors:0 dropped:0 overruns:0 frame:0 TX packets:0 errors:0 dropped:0 overruns:0 carrier:0 collisions:0 txqueuelen:1000 RX bytes:0 TX bytes:0 ......
系统适配
主要是修改 ril 库相关的文件,实现拨号和衔接数据通路。
ril 库移植
-
涉及文件列表
android/hardware/ril/rild/radio.xml android/hardware/ril/rild/rild.rc android/device/softwinner/common/sepolicy/vendor/rild.te android/vendor/aw/public/prebuild/lib/librild/radio_common.mk android/vendor/aw/public/prebuild/lib/librild/lib/lib32/libquectel-ril.so android/vendor/aw/public/prebuild/lib/librild/lib/lib64/libquectel-ril.so
-
构建文件拷贝脚本,将移远的 ril 库以及 apns-conf.xml、ql-ril.conf 文件更新到系统指定目录下
android/vendor/aw/public/prebuild/lib/librild/lib/lib32/libquectel-ril.so android/vendor/aw/public/prebuild/lib/librild/lib/lib64/libquectel-ril.so
-
修改 ril-daemon 服务启动参数
diff --git a/android/hardware/ril/rild/rild.rc b/android/hardware/ril/rild/rild.rc old mode 100644 new mode 100755 index f6beb54..d259206 --- a/android/hardware/ril/rild/rild.rc +++ b/android/hardware/ril/rild/rild.rc @@ -1,5 +1,5 @@ service vendor.ril-daemon /vendor/bin/hw/rild class main - user radio - group radio cache inet misc audio log readproc wakelock + user root + group radio cache inet misc audio sdcard_rw log capabilities BLOCK_SUSPEND NET_ADMIN NET_RAW
-
选择编译 android.hardware.radio@1.0
diff --git a/android/vendor/aw/public/prebuild/lib/librild/radio_common.mk b/android/vendor/aw/public/prebuild/lib/librild/radio_common.mk old mode 100644 new mode 100755 index a7206b6..8ef8bd0 --- a/android/vendor/aw/public/prebuild/lib/librild/radio_common.mk +++ b/android/vendor/aw/public/prebuild/lib/librild/radio_common.mk @@ -5,7 +5,7 @@ RIL_LIB_BASENAME ?= libsoftwinner-ril-$(RIL_LIB_VERSION) # 3G Data Card Packages PRODUCT_PACKAGES += \ - android.hardware.radio@1.4 \ + android.hardware.radio@1.0 \ android.hardware.radio.config@1.1-service \ android.hardware.radio.config@1.1 \ pppd_vendor \
-
修改 hidl 使用 radio 1.0 版本
diff --git a/android/hardware/ril/rild/radio.xml b/android/hardware/ril/rild/radio.xml old mode 100644 new mode 100755 index 866b401..885b150 --- a/android/hardware/ril/rild/radio.xml +++ b/android/hardware/ril/rild/radio.xml @@ -1,12 +1,11 @@ <manifest version="1.0" type="device" target-level="4"> - <hal format="hidl"> - <name>android.hardware.radio</name> - <transport>hwbinder</transport> - <version>1.4</version> - <interface> - <name>IRadio</name> - <instance>slot1</instance> - </interface> - <fqname>@1.2::ISap/slot1</fqname> - </hal> + <hal format="hidl"> + <name>android.hardware.radio</name> + <transport>hwbinder</transport> + <version>1.0</version> + <interface> + <name>IRadio</name> + <instance>slot1</instance> + </interface> + </hal> </manifest>
-
修改 SELinux 权限
diff --git a/android/device/softwinner/common/sepolicy/vendor/rild.te b/android/device/softwinner/common/sepolicy/vendor/rild.te index 8fdddc5..d29b4a2 100755 --- a/android/device/softwinner/common/sepolicy/vendor/rild.te +++ b/android/device/softwinner/common/sepolicy/vendor/rild.te @@ -5,6 +5,7 @@ allow rild vendor_file:file execute_no_trans; allow rild vendor_ril_prop:file { read getattr open }; allow rild vendor_ril_prop:property_service set; allow rild vendor_default_prop:file {open read getattr}; +allow rild rild:packet_socket create_socket_perms_no_ioctl; allow rild vendor_toolbox_exec:file execute_no_trans; allow rild vendor_shell_exec:file execute_no_trans;
完成标志
-
查看系统属性
mercury-demo:/ # getprop | grep ril getprop | grep ril [gsm.version.ril-impl]: [Quectel_Android_RIL_Driver_V3.5.0] [init.svc.vendor.ril-daemon]: [running] [ro.boottime.vendor.ril-daemon]: [9918673186] [ro.radio.noril]: [false] [vendor.rild.libargs]: [-d/dev/ttyUSB2] [vendor.rild.libpath]: [/vendor/lib64/libquectel-ril.so]
-
查看 raido log 信息:logcat -b radio -v time
D/RILD ( 1917): **RIL Daemon Started** D/RILD ( 1917): **RILd param count=3** D/RILD ( 1917): RIL_Init argc = 4 clientId = 0 D/RILC ( 1917): Quectel RIL Version: Quectel_Android_RIL_Driver_V3.5.0 D/RILC ( 1917): libquectel-ril build info: On branch master Your branch is up to date with 'origin/master'. e1552af97d835442b8f39f69118082650eb33081 09/27/22_14:42:59 D/RILC ( 1917): t-prop_value is 2 D/RILC ( 1917): before_dot is 10 D/RILC ( 1917): [ro.build.version.release]: [10.0] D/RILC ( 1917): Android Version: 100, RIL_VERSION: 12 / 12 E/RILC ( 1917): '/vendor/manifest.xml' not exist. D/RILC ( 1917): 1.1</version> D/RILC ( 1917): IRadioVersion 1.1 D/RILC ( 1917): IRadioConfigVersion 1.-1 ...... D/SubscriptionController( 2450): [getPhoneId] asked for default subId=1 D/SubscriptionController( 2450): [getPhoneId] asked for default subId=1 D/SETUP_DATA_CALL( 1917): do_dhcp_request ipaddr=10.192.252.55, prefixLength=24 D/SETUP_DATA_CALL( 1917): ql_set_data_call_info pdp=1, ifname=usb0, iptype=4, ipaddr=10.192.252.55 E/SETUP_DATA_CALL( 1917): ifc_add_address() = -22 D/SETUP_DATA_CALL( 1917): deactivateManually is 0 D/SETUP_DATA_CALL( 1917): usb0 IPv4 Address: 10.192.252.55 D/SETUP_DATA_CALL( 1917): usb0 IPv6 Address: 0:0:fe80::4b:b3ff D/SETUP_DATA_CALL( 1917): usb0 IP Address: 10.192.252.55 D/SETUP_DATA_CALL( 1917): get_local_ip: 10.192.252.55, cost 4 sec
测试验证
-
查看网卡是否出现 usb0 网卡,也检查是否有获取到 ip 地址:
mercury-demo:/ # ifconfig ifconfig eth1 Link encap:Ethernet HWaddr 36:c9:e3:f1:b8:05 Driver sunxi-gmac UP BROADCAST MULTICAST MTU:1500 Metric:1 RX packets:0 errors:0 dropped:0 overruns:0 frame:0 TX packets:5 errors:0 dropped:0 overruns:0 carrier:0 collisions:0 txqueuelen:1000 RX bytes:0 TX bytes:418 Interrupt:66 lo Link encap:Local Loopback inet addr:127.0.0.1 Mask:255.0.0.0 inet6 addr: ::1/128 Scope: Host UP LOOPBACK RUNNING MTU:65536 Metric:1 RX packets:16 errors:0 dropped:0 overruns:0 frame:0 TX packets:16 errors:0 dropped:0 overruns:0 carrier:0 collisions:0 txqueuelen:1 RX bytes:1920 TX bytes:1920 eth0 Link encap:Ethernet HWaddr 36:c9:e3:f1:b8:05 Driver sunxi-gmac UP BROADCAST MULTICAST MTU:1500 Metric:1 RX packets:0 errors:0 dropped:0 overruns:0 frame:0 TX packets:2 errors:0 dropped:0 overruns:0 carrier:0 collisions:0 txqueuelen:1000 RX bytes:0 TX bytes:168 Interrupt:65 usb0 Link encap:Ethernet HWaddr 02:4b:b3:b9:eb:e5 Driver cdc_ether inet addr:10.209.124.163 Bcast:10.209.124.255 Mask:255.255.255.0 inet6 addr: fe80::4b:b3ff:feb9:ebe5/64 Scope: Link UP BROADCAST RUNNING MULTICAST MTU:1500 Metric:1 RX packets:123 errors:0 dropped:0 overruns:0 frame:0 TX packets:175 errors:0 dropped:0 overruns:0 carrier:0 collisions:0 txqueuelen:1000 RX bytes:20877 TX bytes:23570
-
查看是否能正常 ping 通外网:
mercury-demo:/ # ping www.qq.com ping www.qq.com PING ins-r23tsuuf.ias.tencent-cloud.net (101.91.22.57) 56(84) bytes of data. 64 bytes from 101.91.22.57: icmp_seq=1 ttl=48 time=63.5 ms 64 bytes from 101.91.22.57: icmp_seq=2 ttl=48 time=55.8 ms 64 bytes from 101.91.22.57: icmp_seq=3 ttl=48 time=55.1 ms 64 bytes from 101.91.22.57: icmp_seq=4 ttl=48 time=55.8 ms 64 bytes from 101.91.22.57: icmp_seq=5 ttl=48 time=54.9 ms 64 bytes from 101.91.22.57: icmp_seq=6 ttl=48 time=62.9 ms 64 bytes from 101.91.22.57: icmp_seq=7 ttl=48 time=54.8 ms 64 bytes from 101.91.22.57: icmp_seq=8 ttl=48 time=55.4 ms 64 bytes from 101.91.22.57: icmp_seq=9 ttl=48 time=54.8 ms
调试技巧
模组测试
可以先将模组接到 PC 端,使用串口手动下发 AT 指令测试,确定模组可以正常使用。
AT +CPIN?
CPIN: READY
OK
AT+QNWINFO
+QNWINFO:"FDD LTE","46011","LTE BAND 1",100
OK
AT+COPS?
+COPS: 0,0,"CHN-CT",7
OK
AT+QIACT=1
OK
AT+QIACT?
+QIACT: 1,1,1,"10.193.145.135"
OK
AT+QPING=1,"www.baidu.com"
OK
+QPING: 0,"101.91.22.57",32,78,255
+QPING: 0,"101.91.22.57",32,83,255
+QPING: 0,"101.91.22.57",32,80,255
日志提取
启动日志保存,在 data 目录下创建 quectel_debug_log 目录,libquectel-ril.so 会自动生成日志文件。
mkdir /data/quectel_debug_log
mercury-demo:/data/quectel_debug_log $ ls -l
-rw-r--r-- 1 Administrator 197121 788266 12月 30 22:10 1229_144251_logcat.txt
-rw-r--r-- 1 Administrator 197121 37788 12月 30 22:10 1229_144251_radio.txt
-rw-r--r-- 1 Administrator 197121 87888 12月 30 22:10 1229_144251_usbmonlog.txt
-rw-r--r-- 1 Administrator 197121 77156 12月 30 22:10 1229_144316_dmesg_mainLoop.txt