rtl8812驱动分析(一)

来源:互联网 发布:昭大网络教育 编辑:程序博客网 时间:2024/06/06 02:23

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->usbdrv);}

初始化函数主要做的事情:

  1. 打开wifi模块的电源,会根据不同的平台不同的模块进行处理;
  2. 初始化挂起锁,这个后边再分析,猜是防止多线程同时挂起模块时会引起模块问题;
  3. 驱动注册标志位置1;
  4. 注册usb驱动,因为我们的模块接口是USB接口的,所以对系统来说其实就是个USB设备,自然要注册usb驱动。

接着,我们来重点看一下这个注册的usb驱动:

#ifdef CONFIG_RTL8192Cstatic 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驱动中主要提供的接口如下:

  1. probe探测函数
  2. disconnet函数
  3. idtable列表,表示支持的设备的PIDVID信息
  4. 挂起和唤醒函数

在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的过程,以及一些内核线程的创建过程。

0 0
原创粉丝点击