国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 學(xué)院 > 開發(fā)設(shè)計 > 正文

WiFi驅(qū)動框架解析

2019-11-09 13:44:10
字體:
供稿:網(wǎng)友
linux:3.10WiFi芯片:RTL8723接口:SDIO

本文從硬件結(jié)構(gòu)到軟件實現(xiàn)探究Linux中WiFi驅(qū)動的框架。如下圖:

硬件角度:CPUWiFi芯片,以rtl8723為例接口SDIO

軟件角度:1、電源、GPIO:負(fù)責(zé)WiFi模組的電源管理、IO管理2、SDIO:數(shù)據(jù)通道3、WiFi驅(qū)動:負(fù)責(zé)WiFi規(guī)范實現(xiàn)4、Sysfs:/sys/文件系統(tǒng)中提供訪問接口

Linux有非常好的模塊化機制,所以這幾部分作為各自獨立的模塊進(jìn)行注冊,下面從代碼示例的方式看下。

一、電源、GPIO管理模塊

該模塊完成CPU對WiFi模組電源、引腳的初始化、控制等功能。由于Linux采用設(shè)備樹(Device Tree)方式管理硬件設(shè)置,所以第一步就是解析dts文件中的設(shè)置項并進(jìn)行賦值、初始化操作,如:

static struct of_device_id wlan_platdata_of_match[] = {    { .compatible = "wlan-platdata" },    { }};MODULE_DEVICE_TABLE(of, wlan_platdata_of_match);static int wlan_platdata_parse_dt(struct device *dev, struct wifi_moudle *data){	struct device_node *node = dev->of_node;    ret = of_PRoperty_read_string(node, "wifi_chip_type", &strings);	ret = of_property_read_u32(node, "sdio_vref", &value);    of_find_property(node, "keep_wifi_power_on", NULL)    of_find_property(node, "vref_ctrl_enable", NULL)	of_find_property(node, "power_ctrl_by_pmu", NULL);	of_get_named_gpio_flags(node, "WIFI,poweren_gpio", 0, &flags);    of_get_named_gpio_flags(node, "WIFI,reset_gpio", 0, &flags);    of_get_named_gpio_flags(node, "WIFI,host_wake_irq", 0, &flags);    return 0;}除此以外,還需要導(dǎo)出設(shè)置方式,以供其他模塊進(jìn)行需要的設(shè)置,如:

EXPORT_SYMBOL(get_wifi_chip_type);驅(qū)動以平臺驅(qū)動的方式進(jìn)行注冊:

static struct platform_driver wlan_driver = {	.probe = wlan_probe,	.remove = wlan_remove,    .suspend = wlan_suspend,    .resume = wlan_resume,	.driver = {		.name = "wlan-platdata",		.owner = THIS_MODULE,        .of_match_table = of_match_ptr(wlan_platdata_of_match),	},};static int wlan_probe(struct platform_device *pdev){	......	wlan_platdata_parse_dt(&pdev->dev, pdata);	......}static int __init wlan_init(void){    LOG("Enter %s/n", __func__);	return platform_driver_register(&wlan_driver);}module_init(wlan_init);module_exit(wlan_exit);

二、SDIO Host端

CPU集成由SDIO控制器,所以Host端就是對CPU上SDIO控制器的編程、使用,然后把函數(shù)指針賦予Core層(分層思想)。這部分代碼一般存在于drivers/mmc/host/目錄。操作接口比如:

static const struct mmc_host_ops dw_mci_ops = {	.request		= dw_mci_request,	.pre_req		= dw_mci_pre_req,	.post_req		= dw_mci_post_req,	.set_ios		= dw_mci_set_ios,	.get_ro			= dw_mci_get_ro,	.get_cd			= dw_mci_get_cd,	.set_sdio_status	= dw_mci_set_sdio_status,	.hw_reset		= dw_mci_hw_reset,	.enable_sdio_irq	= dw_mci_enable_sdio_irq,	.execute_tuning		= dw_mci_execute_tuning,        .post_tmo		= dw_mci_post_tmo,	#ifdef CONFIG_MMC_DW_ROCKCHIP_SWITCH_VOLTAGE        .start_signal_voltage_switch				= dw_mci_start_signal_voltage_switch,        .card_busy		= dw_mci_card_busy,        #endif};

三、SDIO Client端

WiFi模組自身集成有SDIO控制器,所以這部分完成對WiFi模組上SDIO控制器的編程、使用。針對RTL8723,SDIO Client的代碼實現(xiàn)在其驅(qū)動源碼里。如下:

rtl8723:static struct sdio_drv_priv sdio_drvpriv = {	.r871xs_drv.probe = rtw_drv_init,	.r871xs_drv.remove = rtw_dev_remove,	.r871xs_drv.name = (char*)DRV_NAME,	.r871xs_drv.id_table = sdio_ids,	#if (LINUX_VERSION_CODE >= KERNEL_VERSION(2,6,29)) 	.r871xs_drv.drv = {		.pm = &rtw_sdio_pm_ops,	}	#endif};static int rtw_drv_entry(void){	sdio_drvpriv.drv_registered = _TRUE;	// sdio_register_driver: kernel/drivers/mmc/core/sdio_bus.c	ret = sdio_register_driver(&sdio_drvpriv.r871xs_drv);	return ret;}	int rtl8723_wifi_init_module(void){	return rtw_drv_entry();}late_initcall(rtl8723_wifi_init_module);

四、Sysfs

通過sys文件系統(tǒng),用戶可以讀寫驅(qū)動信息。

static ssize_t wifi_chip_read(struct class *cls, struct class_attribute *attr, char *_buf){	ssize_t count = sprintf(_buf, "%s", "RTL8723BS");	printk("Current WiFi chip is RTL8723BS./n");	return count;}static ssize_t wifi_power_write(struct class *cls, struct class_attribute *attr, const char *_buf, size_t _count){	int poweren = 0;    poweren = simple_strtol(_buf, NULL, 10);    if(poweren > 0) {        wifi_power(1);    } else {        wifi_power(0);    }	return _count;}static struct class *wifi_class = NULL;// 生成class_attr_chipstatic CLASS_ATTR(chip, 0664, wifi_chip_read, NULL);// 生成class_attr_powerstatic CLASS_ATTR(power, 0660, NULL, wifi_power_write);int wifi_sysif_init(void){    int ret;    wifi_class = class_create(THIS_MODULE, "rtlwifi");        ret = class_create_file(wifi_class, &class_attr_chip);    ret = class_create_file(wifi_class, &class_attr_power);    return 0;}void wifi_sysif_exit(void){    // need to remove the sys files and class    class_remove_file(wifi_class, &class_attr_chip);    class_remove_file(wifi_class, &class_attr_power);            wifi_class = NULL;}module_init(wifi_sysif_init);module_exit(wifi_sysif_exit);該模塊注冊后,將出現(xiàn)/sys/class/rtlwifi目錄,且目錄下含有/sys/class/rtlwifi/chip、/sys/class/rtlwifi/power兩個文件,cat chip文件將會得到RTL8723BS,而讀寫power文件能夠獲取、設(shè)置WiFi模組供電狀態(tài)。


發(fā)表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發(fā)表
主站蜘蛛池模板: 湟源县| 柳河县| 乐昌市| 陇西县| 浑源县| 晋城| 苗栗市| 横山县| 龙泉市| 二手房| 庐江县| 黔江区| 彭阳县| 西贡区| 辽宁省| 丹寨县| 文山县| 普定县| 桃园县| 岚皋县| 岳池县| 朔州市| 东阿县| 特克斯县| 霍山县| 四会市| 津南区| 双辽市| 茶陵县| 凯里市| 博罗县| 蓝山县| 治多县| 阿巴嘎旗| 鄂伦春自治旗| 甘泉县| 朝阳市| 西乡县| 纳雍县| 达州市| 舞阳县|