rtl8812驱动分析(一)
2015-06-20 12:08
1281 查看
wifi驱动的入口->os_dep/linux/usb_intf.c
insmod 8188.ko和rmmod 8188.ko时分别调用的是:module_init(rtw_drv_entry);
module_exit(rtw_drv_halt);
所以,找到这两个函数,就分别找到了初始化和退出函数
初始化:
static int __init rtw_drv_entry(void) { #ifdef CONFIG_PLATFORM_RTK_DMP u32 tmp; tmp=readl((volatile unsigned int*)0xb801a608); tmp &= 0xffffff00; tmp |= 0x55; writel(tmp,(volatile unsigned int*)0xb801a608);//write dummy register for 1055 #endif #ifdef CONFIG_PLATFORM_ARM_SUNxI #ifndef CONFIG_RTL8723A int ret = 0; /* ----------get usb_wifi_usbc_num------------- */ ret = script_parser_fetch("usb_wifi_para", "usb_wifi_usbc_num", (int *)&usb_wifi_host, 64); if(ret != 0){ printk("ERR: script_parser_fetch usb_wifi_usbc_num failed\n"); ret = -ENOMEM; return ret; } printk("sw_usb_enable_hcd: usbc_num = %d\n", usb_wifi_host); sw_usb_enable_hcd(usb_wifi_host); #endif //CONFIG_RTL8723A #endif //CONFIG_PLATFORM_ARM_SUNxI #ifdef CONFIG_PLATFORM_ARM_SUN6I script_item_value_type_e type; type = script_get_item("wifi_para", "wifi_usbc_id", &item); if(SCIRPT_ITEM_VALUE_TYPE_INT != type){ printk("ERR: script_get_item wifi_usbc_id failed\n"); return -ENOMEM; } printk("sw_usb_enable_hcd: usbc_num = %d\n", item.val); wifi_pm_power(1); mdelay(10); sw_usb_enable_hcd(item.val); #endif //CONFIG_PLATFORM_ARM_SUN6I RT_TRACE(_module_hci_intfs_c_,_drv_err_,("+rtw_drv_entry\n")); DBG_871X(DRV_NAME " driver version=%s\n", DRIVERVERSION); DBG_871X("build time: %s %s\n", __DATE__, __TIME__); #if (LINUX_VERSION_CODE >= KERNEL_VERSION(2,6,24)) //console_suspend_enabled=0; #endif rtw_suspend_lock_init(); usb_drv->drv_registered = _TRUE; return usb_register(&usb_drv->usb 4000 drv); }
初始化函数主要做的事情:
打开wifi模块的电源,会根据不同的平台不同的模块进行处理;
初始化挂起锁,这个后边再分析,猜是防止多线程同时挂起模块时会引起模块问题;
驱动注册标志位置1;
注册usb驱动,因为我们的模块接口是USB接口的,所以对系统来说其实就是个USB设备,自然要注册usb驱动。
接着,我们来重点看一下这个注册的usb驱动:
#ifdef CONFIG_RTL8192C static struct usb_device_id rtl8192c_usb_id_tbl[] ={ RTL8192C_USB_IDS {} /* Terminating entry */ }; struct rtw_usb_drv rtl8192c_usb_drv = { .usbdrv.name = (char*)"rtl8192cu", .usbdrv.probe = rtw_drv_init, .usbdrv.disconnect = rtw_dev_remove, .usbdrv.id_table = rtl8192c_usb_id_tbl, .usbdrv.suspend = rtw_suspend, .usbdrv.resume = rtw_resume, #if (LINUX_VERSION_CODE > KERNEL_VERSION(2, 6, 22)) .usbdrv.reset_resume = rtw_resume, #endif #ifdef CONFIG_AUTOSUSPEND .usbdrv.supports_autosuspend = 1, #endif #if (LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 19)) .usbdrv.drvwrap.driver.shutdown = rtw_dev_shutdown, #else .usbdrv.driver.shutdown = rtw_dev_shutdown, #endif }; static struct rtw_usb_drv *usb_drv = &rtl8192c_usb_drv; #endif /* CONFIG_RTL8192C */
我们的模块是8812CUS,但是不知道为何是8192c的配置选项?留着疑问,后边解决。
注册的usb驱动中主要提供的接口如下:
probe探测函数
disconnet函数
idtable列表,表示支持的设备的PIDVID信息
挂起和唤醒函数
在idtables中我们可以看到 RTL8192C_USB_IDS的定义,找到它,我们发现,我们的8812CUS模块也在其中(通过查看PID和VID发现我们的模块型号为PID8176,即列表中的第二行):
#define RTL8192C_USB_IDS \ /*=== Realtek demoboard ===*/ \ {USB_DEVICE(USB_VENDER_ID_REALTEK, 0x8191)},/* Default ID */ \ /****** 8188CUS ********/ \ {USB_DEVICE(USB_VENDER_ID_REALTEK, 0x8176)},/* 8188cu 1*1 dongole */ \ {USB_DEVICE(USB_VENDER_ID_REALTEK, 0x8170)},/* 8188CE-VAU USB minCard */ \ {USB_DEVICE(USB_VENDER_ID_REALTEK, 0x817E)},/* 8188CE-VAU USB minCard */ \ {USB_DEVICE(USB_VENDER_ID_REALTEK, 0x817A)},/* 8188cu Slim Solo */ \
这个驱动insmod后,注册到USB驱动链上,当usb设备接入,由usb子系统枚举后,获取设备的描述符,最后通过描述符中的PID和VID信息为其匹配驱动,通过遍历usb驱动链上的id列表的方式找到支持该设备的驱动,最后在这个列表中找到匹配,并将设备和这个驱动绑定。
后续继续分析wifi驱动的probe的过程,以及一些内核线程的创建过程。
相关文章推荐
- mysql中int、bigint、smallint 和 tinyint的区别与长度
- mysql #1062 –Duplicate entry '1' for key 'PRIMARY'
- 深入解析unsigned int 和 int
- asp Fix、Int、Round、CInt函数使用说明
- short与int转换的小例子
- 无法从 int? 转换为 int 运行时出现错误
- mysql中int、bigint、smallint 和 tinyint的区别详细介绍
- mysql error:#1062 Duplicate entry ‘***′ for key 1问题解决方法
- Javascript将string类型转换int类型
- php下intval()和(int)转换使用与区别
- C#中把字符串String转换为整型Int的小例子
- Last_Errno: 1062,Last_Error: Error Duplicate entry
- SQLServer中bigint转int带符号时报错问题解决方法
- sqlserver中将varchar类型转换为int型再进行排序的方法
- js中string转int把String类型转化成int类型
- Java中对AtomicInteger和int值在多线程下递增操作的测试
- Java中关于int和Integer的区别详解
- python类型强制转换long to int的代码
- char转int
- operator int-C++