歡迎來到Linux教程網
Linux教程網
Linux教程網
Linux教程網
Linux教程網 >> Linux編程 >> Linux編程 >> Android Wifi驅動--底層

Android Wifi驅動--底層

日期:2017/3/1 11:08:57   编辑:Linux編程

由於在這個項目中,WIFI模塊是采用SDIO總線來控制的,所以先記錄下CLIENT DRIVER的SDIO部分的結構,這部分的SDIO分為三層:SdioDrv、SdioAdapter、SdioBusDrv。其中SdioBusDrv是Client Driver中SDIO與WIFI模塊的接口,SdioAdapter是SdioDrv和SdioBusDrv之間的適配層,SdioDrv是Client Driver中SDIO與LINUX KERNEL中的MMC SDIO的接口。這三部分只需要關注一下SdioDrv就可以了,另外兩層都只是對它的封裝罷了。

在SdioDrv中提供了這幾個功能:

(1)static struct sdio_driver tiwlan_sdio_drv = {
.probe = tiwlan_sdio_probe,
.remove = tiwlan_sdio_remove,
.name = "sdio_tiwlan",
.id_table = tiwl12xx_devices,
};


(2)int sdioDrv_EnableFunction(unsigned int uFunc)


(3)int sdioDrv_EnableInterrupt(unsigned int uFunc)

(4)SDIO的讀寫,實際是調用了MMC\Core中的 static int mmc_io_rw_direct_host()功能。

SDIO功能部分簡單了解下就可以,一般HOST部分芯片廠商都會做好。我的主要任務還是WIFI模塊。

首先從WIFI模塊的入口函數wlanDrvIf_ModuleInit()看起,這裡調用了wlanDrvIf_Create()。

代碼主體部分:

static int wlanDrvIf_Create (void)
{
TWlanDrvIfObj *drv; //這個結構體為代表設備,包含LINUX網絡設備結構體net_device

pDrvStaticHandle = drv; /* save for module destroy */

drv->pWorkQueue = create_singlethread_workqueue (TIWLAN_DRV_NAME);//創建了工作隊列

/* Setup driver network interface. */
rc = wlanDrvIf_SetupNetif (drv); //這個函數超級重要,後面詳細的看

drv->wl_sock = netlink_kernel_create( NETLINK_USERSOCK, 0, NULL, NULL, THIS_MODULE );


// 創建了接受wpa_supplicant的SOCKET接口


/* Create all driver modules and link their handles */
rc = drvMain_Create (drv,
&drv->tCommon.hDrvMain,
&drv->tCommon.hCmdHndlr,
&drv->tCommon.hContext,
&drv->tCommon.hTxDataQ,
&drv->tCommon.hTxMgmtQ,
&drv->tCommon.hTxCtrl,
&drv->tCommon.hTWD,
&drv->tCommon.hEvHandler,
&drv->tCommon.hCmdDispatch,
&drv->tCommon.hReport,
&drv->tCommon.hPwrState);

/*
* Initialize interrupts (or polling mode for debug):
*/

/* Normal mode: Interrupts (the default mode) */
rc = hPlatform_initInterrupt (drv, (void*)wlanDrvIf_HandleInterrupt);

return 0;

}

在調用完wlanDrvIf_Create()這個函數後,實際上WIFI模塊的初始化就結束了,下面分析如何初始化的。先看wlanDrvIf_SetupNetif (drv)這個函數的主體,

static int wlanDrvIf_SetupNetif (TWlanDrvIfObj *drv)
{
struct net_device *dev;
int res;

/* Allocate network interface structure for the driver */
dev = alloc_etherdev (0);//申請LINUX網絡設備
if (dev == NULL)

/* Setup the network interface */
ether_setup (dev);//建立網絡接口 ,這兩個都是LINUX網絡設備驅動的標准函數

dev->netdev_ops = &wlan_netdev_ops;

/* Initialize Wireless Extensions interface (WEXT) */
wlanDrvWext_Init (dev);

res = register_netdev (dev);

/* Setup power-management callbacks */
hPlatform_SetupPm(wlanDrvIf_Suspend, wlanDrvIf_Resume, pDrvStaticHandle);

}

注意,在這裡初始化了wlanDrvWext_Inti(dev),這就說明wpa_supplicant與Driver直接的聯系是走的WEXT這條路。也就是說event的接收,處理也應該是在WEXT部分來做的,確定這個,剩下的工作量頓減三分之一,哈哈哈。後面還注冊了網絡設備dev。而在wlan_netdev_ops中定義的功能如下:

static const struct net_device_ops wlan_netdev_ops = {
.ndo_open = wlanDrvIf_Open,
.ndo_stop = wlanDrvIf_Release,
.ndo_do_ioctl = NULL,

.ndo_start_xmit = wlanDrvIf_Xmit,
.ndo_get_stats = wlanDrvIf_NetGetStat,
.ndo_validate_addr = NULL,

};

功能一看名字就知道了,不說了,這幾個對應的都是LINUX網絡設備驅動都有的命令字,詳見《LINUX設備驅動開發詳解》第十六章(見http://www.linuxidc.com/Linux/2011-07/38211.htm)。

在這之後,又調用了rc =drvMain_CreateI。

在這個函數裡完成了相關模塊的初始化工作。具體不說了。接下來就是等待Android上層發送來的事件了。

Copyright © Linux教程網 All Rights Reserved