日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 >

wifi详解(三)

發布時間:2025/5/22 45 豆豆
生活随笔 收集整理的這篇文章主要介紹了 wifi详解(三) 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

1????????WLAN驅動結構介紹

1.1??????SDIO驅動

在drivers/mmc下面是mmc卡,SD卡和SDIO卡驅動部分,其中包括host驅動,card驅動和core部分,由于網絡接口卡掛接在SDIO總線上,所以在此之前我們先看一下SDIO的驅動結構。其驅動在drivers/mmc目錄下的結構為:

?

|-- mmc

|?? |-- card

|?? |-- core

|?? |-- host

?

主要關注的目錄是core目錄,這個目錄是真個驅動的核心目錄,是媒體卡的通用代碼部分,包括core.c,host.c和sdio.c等。CORE 層完成了不同協議和規范的實現,并為HOST 層的驅動提供了接口函數,該目錄完成sdio總線的注冊操作,相應的ops操作,以及支持mmc的代碼。詳細的情況將在函數接口部分詳細討論。

Host目錄是不同平臺根據平臺的特性而編寫的host驅動。

1.2??????Boardcom無線通訊芯片

1.2.1?????? 概述

全球有線和無線通信半導體市場的領導者Broadcom(博通)公司(Nasdaq:BRCM)宣布,推出最新無線組合芯片BCM4330,該芯片可支持更多媒體形式和數據應用,且不會增大智能手機、平板電腦及其他移動設備的尺寸或縮短其電池壽命。BCM4330在單個芯片上集成了業界領先的Broadcom 802.11n Wi-Fi、藍牙和FM無線技術,與分立式半導體器件組成的解決方案相比,在成本、尺寸、功耗和性能上有顯著優勢,是移動設備的理想選擇。

BCM4330采用了新的Wi-Fi和藍牙標準,可支持新的、令人振奮的應用。例如,Broadcom BCM4330是業界第一款經過藍牙4.0標準認證的組合芯片解決方案, 集成了藍牙低功耗(BLE)標準。該標準使藍牙技術能以超低功耗運行,因此BCM4330非常適用于需要很長電池壽命的系統,如無線傳感器、醫療和健身監控設備等。BCM4330還支持Wi-Fi Direct?和藍牙高速(HS)標準,因此采用BCM4330的移動設備能直接相互通信,而不必先連接到接入點、成為傳統網絡的一部分,從而為很多無線設備之間新的應用和使用模式創造了機會。

Broadcom一直支持所有主流的操作系統(OS)平臺,如MicrosoftWindows和Windows Phone、Google Chrome、Android等等,而且不僅是BCM4330,所有藍牙、WLAN和GPS芯片組都提供這樣的支持。

1.2.2?????? 源碼

Bcm4330驅動源碼一般被廠商單獨提供,如果要在開發的LINUX系統中(當然它還支持多種平臺)使用該源碼,可以添加到linux kernel源碼樹里,也可以單獨組織存放,可以直接編譯到kernel,也可以編譯成模塊,然后再系統啟動的流程中或其他適當的實際加載到kernel中,一般建議單獨組織并編譯成模塊在需要的時候加載如kernel。

|-- src

|?? |-- bcmsdio

|?? |-- dhd

|?? |--dongle

|?? |--include

|?? |-- shared

|?? |-- wl

?

這里主要內容到bcmsdio,dhd和wl三個目錄下,bcm4330驅動的入口在dhd/sys/dhd_linux.c文件中的dhd_module()函數,設備的初始化和相關驅動注冊都從這里開始,

1.3??????詳細接口及代碼分析

1.3.1??????WIFI驅動流程分析

??? 以boardcom bcm4329芯片驅動為例,相應的函數流程圖如下:???????

???????????????????????

?

1.3.2?????? WIFI設備注冊流程

?

Platform_driver_register(wifi_device[_legacy])的調用將wifi_device[_legacy]驅動注冊到系統中,wifi_device_legacy是為了兼容老版本的驅動。

Path: wl/sys/wl_android.c

Static struct Platform_driver?wifi_device?={

???????? .probe???? =???? wifi_probe

???????? .remove?? =???? wifi_remove

???????? .suspend? =???? wifi_supend

???????? .resume?? =???? wifi_resume

???????? .driver???? =???? {

???????? .name????? =?????“bcmdhd_wlan”

}

}

?

Static struct Platform_driver?wifi_device_legacy?={

???????? .probe???? =???? wifi_probe

???????? .remove?? =???? wifi_remove

???????? .suspend? =???? wifi_supend

???????? .resume?? =???? wifi_resume

???????? .driver???? =???? {

???????? .name????? =?????“bcm4329_wlan”

}

}

上面的展示了wifi平臺設備驅動的注冊過程,那么在平臺相關的代碼區應該有wifi作為平臺設備被初始化和注冊的地方:

Path: kernel/arch/arm/mach-msm/msm_

static struct resource mahimahi_wifi_resources[] = {

??????? [0] = {

??????????????? .name?????????? = "bcm4329_wlan_irq",

??????????????? .start????????? =MSM_GPIO_TO_INT(MAHIMAHI_GPIO_WIFI_IRQ),

??????????????? .end??????????? = MSM_GPIO_TO_INT(MAHIMAHI_GPIO_WIFI_IRQ),

??????????????? .flags????????? = IORESOURCE_IRQ |IORESOURCE_IRQ_HIGHLEVEL | IORESOURCE_IRQ_SHAREABLE,

??????? },

};

?

static structwifi_platform_data mahimahi_wifi_control = {

????????.set_power????? = mahimahi_wifi_power,

??? ????.set_reset????? = mahimahi_wifi_reset,

????????.set_carddetect = mahimahi_wifi_set_carddetect,

??????? .mem_prealloc?? = mahimahi_wifi_mem_prealloc,

};

?

static struct platform_device mahimahi_wifi_device = {

????????.name?????????? = "bcm4329_wlan",

????????.id???????????? = 1,

????????.num_resources? = ARRAY_SIZE(mahimahi_wifi_resources),

????????.resource?????? = mahimahi_wifi_resources,

????????.dev??????????? = {

???????????????.platform_data = &mahimahi_wifi_control,

????????},

};

上面是對wifi_device設備的初始化,下面是對該設備的注冊:

static int __initmahimahi_wifi_init(void)

{

??????? int ret;

?

??????? if (!machine_is_mahimahi())

??????????????? return 0;

?

??????? printk("%s: start\n",__func__);

???????mahimahi_wifi_update_nvs("sd_oobonly=1\r\n", 0);

???????mahimahi_wifi_update_nvs("btc_params70=0x32\r\n", 1);

??????? mahimahi_init_wifi_mem();

????????ret = platform_device_register(&mahimahi_wifi_device);

??????? return ret;

}

?

late_initcall(mahimahi_wifi_init); ?????//表明在系統啟動的后期會自動調用加載該模塊

這樣,通過上面的初始化和注冊流程,wifi設備作為平臺設備和驅動就可以握手成功了,這里的平臺驅動只是對wifi設備的簡單管理,如對wifi設備的掛起和恢復等操作了。但是在wifi設備初始化之前是不能夠被掛起和恢復的,那么wifi設備是如何初始化的呢?

Path: wl/sys/wl_android.c

static int wifi_probe(structplatform_device *pdev)

{

????????struct wifi_platform_data *wifi_ctrl =

????????????????(structwifi_platform_data *)(pdev->dev.platform_data);

?

??????? DHD_ERROR(("## %s\n",__FUNCTION__));

????????wifi_irqres = platform_get_resource_byname(pdev,IORESOURCE_IRQ, "bcmdhd_wlan_irq");

??????? if (wifi_irqres == NULL)

??????????????? wifi_irqres =platform_get_resource_byname(pdev,

??????????????????????? IORESOURCE_IRQ,"bcm4329_wlan_irq");

????????wifi_control_data = wifi_ctrl;

?

????????wifi_set_power(1,0);?? /* Power On */

???????wifi_set_carddetect(1); /* CardDetect (0->1) */

?

??????? up(&wifi_control_sem);

??????? return 0;

}

這是wifi平臺設備驅動注冊時成功匹配wifi設備后調用的函數wifi_probe(),它的主要工作就是從wifi設備中獲取終端資源,并獲取wifi_platform_data類型結構賦予wifi_control_data變量,這一步很重要,下面就可以看出了它的重要性。然后調用wifi_set_power和wifi_set_carddetect函數給wifi芯片上電并檢測。

int wifi_set_power(int on, unsignedlong msec)

{

??????? DHD_ERROR(("%s = %d\n",__FUNCTION__, on));

??????? if (wifi_control_data &&wifi_control_data->set_power) {

????????????????wifi_control_data->set_power(on);

??????? }

??????? if (msec)

??????????????? msleep(msec);

??????? return 0;

}

Wifi_set_power函數中調用wifi_control_data->set_power(on),wifi_control_data就是剛才說的那個重要變量,注意它是從wifi_device平臺設備的wifi_platform_data獲取的,那么看看上面的wifi_device初始化的代碼:

static struct platform_device mahimahi_wifi_device = {

??????? .name?????????? = "bcm4329_wlan",

??????? .id???????????? = 1,

??????? .num_resources? = ARRAY_SIZE(mahimahi_wifi_resources),

??????? .resource?????? = mahimahi_wifi_resources,

??????? .dev??????????? = {

??????????????? .platform_data =&mahimahi_wifi_control,

??????? },

};

?

static struct wifi_platform_datamahimahi_wifi_control= {

??????? .set_power????? =?mahimahi_wifi_power,

??????? .set_reset????? = mahimahi_wifi_reset,

??????? .set_carddetect =?mahimahi_wifi_set_carddetect,

??????? .mem_prealloc?? = mahimahi_wifi_mem_prealloc,

};

所以它實際調用的是mahimahi_wifi_power函數,該函數的定義在kernel/arch/arm /mach-msm/board-mahimahi-mmc.c之中:

int mahimahi_wifi_power(int on)

{

???????printk("%s: %d\n", __func__, on);

?

??????? if (on) {

???????????????config_gpio_table(wifi_on_gpio_table,

?????????????????? ???????????????ARRAY_SIZE(wifi_on_gpio_table));

???????????????mdelay(50);

??????? } else {

???????????????config_gpio_table(wifi_off_gpio_table,

?????????????????????????????????ARRAY_SIZE(wifi_off_gpio_table));

??????? }

?

??????? mdelay(100);

??????? gpio_set_value(MAHIMAHI_GPIO_WIFI_SHUTDOWN_N,on); /* WIFI_SHUTDOWN */

??????? mdelay(200);

?

???????mahimahi_wifi_power_state = on;

??????? return 0;

}

調用gpio_set_value操作wifi芯片,給wifi芯片上電。那么來看看wifi_set_ carddetect函數究竟干了什么:

Path:wl/sys/wl_android.c

static int wifi_set_carddetect(int on)

{

???????DHD_ERROR(("%s = %d\n", __FUNCTION__, on));

??????? if(wifi_control_data && wifi_control_data->set_carddetect) {

???????????????wifi_control_data->set_carddetect(on);

??????? }

??????? return 0;

}

同樣會調用wifi_device的mahimahi_wifi_set_carddetect函數:

Path:kernel/arch/arm/mach-msm/board-mahimahi-mmc.c

int mahimahi_wifi_set_carddetect(int val)

{

???????pr_info("%s: %d\n", __func__, val);

???????mahimahi_wifi_cd = val;

??????? if(wifi_status_cb) {

????????????????wifi_status_cb(val,wifi_status_cb_devid);

??????? } else

???????????????pr_warning("%s: Nobody to notify\n", __func__);

??????? return 0;

}

Wifi_status_cb代碼:

static int mahimahi_wifi_status_register(

??????????????????????? void (*callback)(intcard_present, void *dev_id),

??????????????????????? void *dev_id)

{

??????? if (wifi_status_cb)

??????????????? return -EAGAIN;

????????wifi_status_cb = callback;

?? ?????wifi_status_cb_devid = dev_id;

??????? return 0;

}

?

static unsigned intmahimahi_wifi_status(struct device *dev)

{

??????? return mahimahi_wifi_cd;

}

?

static structmmc_platform_data mahimahi_wifi_data = {

??????? .ocr_mask?????????????? = MMC_VDD_28_29,

??????? .built_in?????????????? = 1,

??????? .status???????????????? = mahimahi_wifi_status,

????????.register_status_notify= mahimahi_wifi_status_register,

??????? .embedded_sdio????????? = &mahimahi_wifi_emb_data,

};

由上面代碼;不難看出會有個地方調用mahimahi_wifi_status_register設置wifi_status_cb這個回調函數,可以跟蹤這個mahimahi_wifi_data結構體,來看看它被傳遞給了誰:

int msm_add_sdcc(unsigned intcontroller, struct mmc_platform_data *plat,

???????????????? unsigned int stat_irq,unsigned long stat_irq_flags);

?

int __initmahimahi_init_mmc(unsigned int sys_rev, unsigned debug_uart)

{

??????? ……

?

????????msm_add_sdcc(1, &mahimahi_wifi_data, 0, 0);

?

?????? ……

??????? if (system_rev > 0)

??????????????? msm_add_sdcc(2,&mahimahi_sdslot_data, 0, 0);

??????? else {

?? ?????????????mahimahi_sdslot_data.status =mahimahi_sdslot_status_rev0;

???????????????mahimahi_sdslot_data.register_status_notify = NULL;

???????????????set_irq_wake(MSM_GPIO_TO_INT(MAHIMAHI_GPIO_SDMC_CD_REV0_N), 1);

????????????????msm_add_sdcc(2, &mahimahi_sdslot_data,

???????? ……

}

可以跟蹤到這里Path:kernel/arch/arm/mach-msm/devices-msm7x30.c

struct platform_device msm_device_sdc1 = {

????????.name?????????? = "msm_sdcc",

????????.id???????????? = 1,

????????.num_resources? = ARRAY_SIZE(resources_sdc1),

????????.resource?????? = resources_sdc1,

????????.dev??????????? = {

???????????????.coherent_dma_mask????? =0xffffffff,

????????},

};

?

struct platform_device msm_device_sdc2 = {

????????.name?????????? = "msm_sdcc",

????????.id???????????? = 2,

????????.num_resources? = ARRAY_SIZE(resources_sdc2),

????????.resource?????? = resources_sdc2,

????????.dev??????????? = {

???????????????.coherent_dma_mask????? =0xffffffff,

????????},

};

?

struct platform_devicemsm_device_sdc3 = {

??????? .name?????????? = "msm_sdcc",

??????? .id???????????? = 3,

??????? .num_resources? = ARRAY_SIZE(resources_sdc3),

??????? .resource?????? = resources_sdc3,

??????? .dev??????????? = {

??????????????? .coherent_dma_mask????? = 0xffffffff,

??????? },

};

?

struct platform_device msm_device_sdc4= {

??????? .name?????????? = "msm_sdcc",

??????? .id???????????? = 4,

??????? .num_resources? = ARRAY_SIZE(resources_sdc4),

??????? .resource?????? = resources_sdc4,

??????? .dev??????????? = {

?????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????439,2-16????? 62%

??????????????? .coherent_dma_mask????? = 0xffffffff,

??????? },

};

?

static struct platform_device *msm_sdcc_devices[] __initdata = {

??????? &msm_device_sdc1,

???? ???&msm_device_sdc2,

??????? &msm_device_sdc3,

??????? &msm_device_sdc4,

};

?

int __initmsm_add_sdcc(unsigned int controller, struct mmc_platform_data *plat,

??????????????????????? unsigned int stat_irq,unsigned long stat_irq_flags)

{

??????? ……

?

?????? ?pdev =msm_sdcc_devices[controller-1]; //因為傳過來的controller是1,所以下面注冊的是第一個平臺設備

????????pdev->dev.platform_data= plat;??//被傳遞給平臺設備的platform_data

?

??????? res =platform_get_resource_byname(pdev, IORESOURCE_IRQ, "status_irq");

??????? if (!res)

?????????????? ?return -EINVAL;

??????? else if (stat_irq) {

??????????????? res->start = res->end =stat_irq;

??????????????? res->flags &=~IORESOURCE_DISABLED;

??????????????? res->flags |=stat_irq_flags;

??????? }

?

????????return platform_device_register(pdev); //如上所述

}

那么這個平臺設備是什么呢,就是sd卡控制器,也就是前面說的host驅動所驅動的主機控制設備。

Path: drivers/mmc/host/msm_sdcc.c

static struct platform_drivermsmsdcc_driver = {

??????? .probe????????? =?msmsdcc_probe,

??????? .suspend??????? = msmsdcc_suspend,

??????? .resume???????? = msmsdcc_resume,

??????? .driver???????? = {

??????????????? .name?? =?"msm_sdcc",

??????? },

};

?

static int __initmsmsdcc_init(void)

{

????????return platform_driver_register(&msmsdcc_driver);

}

驅動成功匹配設備后,調用probe函數:

static int

msmsdcc_probe(structplatform_device *pdev)

{

......

if (stat_irqres &&!(stat_irqres->flags & IORESOURCE_DISABLED)) {

……

??????? } else if(plat->register_status_notify) {

????????????????plat->register_status_notify(msmsdcc_status_notify_cb,host);

??????? } else if (!plat->status)

......

}

msmsdcc_status_notify_cb調用msmsdcc_check_status函數:

msmsdcc_status_notify_cb(intcard_present, void *dev_id)

{

??????? struct msmsdcc_host *host = dev_id;

?

??????? printk(KERN_DEBUG "%s:card_present %d\n", mmc_hostname(host->mmc),

?????????????? card_present);

????????msmsdcc_check_status((unsigned long) host);

}

msmsdcc_check_status調用mmc_detect_change函數:

static void

msmsdcc_check_status(unsignedlong data)

{

??????? ……

??????? if (status ^ host->oldstat) {

??????????????? pr_info("%s: Slot statuschange detected (%d -> %d)\n",

?? ?????????????????????mmc_hostname(host->mmc),host->oldstat, status);

??????????????? if (status &&!host->plat->built_in)

????????????????????????mmc_detect_change(host->mmc, (5 * HZ) / 2);

??????????????? else

????????????????????????mmc_detect_change(host->mmc, 0);

??????? }

?

??????? host->oldstat = status;

?

out:

??????? if (host->timer.function)

??????????????? mod_timer(&host->timer,jiffies + HZ);

}

可以看到mmc_detect_change被調用了,這個函數觸發了一個延時工作:

void mmc_detect_change(structmmc_host *host, unsigned long delay)

{

……

?

????????mmc_schedule_delayed_work(&host->detect, delay);

}

這個時候它會在delay時間后,執行host->detect延時工作對應的函數,在host驅動注冊并匹配設備成功后執行的probe函數里,會調用mmc_alloc_host動態創建一個mmc_host:

msmsdcc_probe(structplatform_device *pdev)

{

......

/*

???????? * Setup our host structure

???????? */

?

????????mmc = mmc_alloc_host(sizeof(struct msmsdcc_host),&pdev->dev);

??????? if (!mmc) {

??????????????? ret = -ENOMEM;

????????? ??????goto out;

??????? }

......

}

mmc_alloc_host初始化工作入口:

struct mmc_host*mmc_alloc_host(int extra, struct device *dev)

{

......

INIT_DELAYED_WORK(&host->detect, mmc_rescan);

......

}

mmc_rescan是core.c中一個很重要的函數,它遵照 SDIO 卡協議的 SDIO 卡啟動過程,包括了非激活模式、卡識別模式和數據傳輸模式三種模式共九種狀態的轉換,你需要參照相關規范來理解。

void mmc_rescan(structwork_struct *work)

{

??????? struct mmc_host *host =

??????????????? container_of(work, structmmc_host, detect.work);

......

?????? ?mmc_power_up(host);

????????sdio_reset(host);

????????mmc_go_idle(host);

?

???????mmc_send_if_cond(host, host->ocr_avail);

?

??????? /*

???????? * First we search for SDIO...

???????? */

????????err = mmc_send_io_op_cond(host, 0, &ocr);

??????? if (!err) {

????????????????if (mmc_attach_sdio(host, ocr))

????????????????????? ??mmc_power_off(host);

??????????????? extend_wakelock = 1;

??????????????? goto out;

??????? }

......

}

這個mmc_attach_sdio函數很重要,它是SDIO卡的初始化的起點,主要工作包括:匹配SDIO卡的工作電壓,分配并初始化mmc_card結構,然后注冊mmc_card到系統中:

/*

?* Starting point for SDIO card init.

?*/

int mmc_attach_sdio(structmmc_host *host, u32 ocr)

{

??????? ……

?

????????mmc_attach_bus(host,&mmc_sdio_ops);? //初始化host的bus_ops

?

?????? ……

?

????????host->ocr = mmc_select_voltage(host, ocr);?//匹配SDIO卡工作電壓

?

??????? ……

?

??????? /*

???????? * Detect and init the card.

???????? */

????????err = mmc_sdio_init_card(host, host->ocr, NULL, 0);//檢測,分配初始化mmc_card

??????? if (err)

??????????????? goto err;

??????? card = host->card;

/*

???????? * If needed, disconnect card detectionpull-up resistor.

???????? */

??????? err = sdio_disable_cd(card);

??????? if (err)

??????????????? goto remove;

?

??????? /*

???????? *?Initialize (but don't add) all present functions.

???????? */

????????for (i = 0; i < funcs; i++, card->sdio_funcs++) {

#ifdef CONFIG_MMC_EMBEDDED_SDIO

????????????????if(host->embedded_sdio_data.funcs) {

???????????????????????struct sdio_func *tmp;

?

????????????????????????tmp = sdio_alloc_func(host->card);

????????????????????????if(IS_ERR(tmp))

???????????????????????????????goto remove;

???????????????????????tmp->num = (i + 1);

???????????????????????card->sdio_func[i] = tmp;

???????????????????????tmp->class = host->embedded_sdio_data.funcs[i].f_class;

???????????????????????tmp->max_blksize = host->embedded_sdio_data.funcs[i].f_maxblksize;

???????????????????????tmp->vendor = card->cis.vendor;

???????????????????????tmp->device = card->cis.device;

????????????????} else {

#endif

??????????????????????? err =sdio_init_func(host->card, i + 1);

??????????????????????? if (err)

??????????????????????????????? goto remove;

#ifdefCONFIG_MMC_EMBEDDED_SDIO

??????????????? }

#endif

??????? }

?

??????? mmc_release_host(host);

?

??????? /*

???????? * First add the card to the drivermodel...

???????? */

????????err = mmc_add_card(host->card);? ?? //添加mmc_card

??????? if (err)

??????????????? goto remove_added;

?

??????? /*

???????? * ...then the SDIO functions.

???????? */

??????? for (i = 0;i < funcs;i++) {

????????????????err =sdio_add_func(host->card->sdio_func[i]);??????????????//將sdio_func加入系統

?????????? ?????if (err)

?????????????????????? goto remove_added;

??????? }

?

??????? return 0;

......

}

這樣,SDIO卡已經初始化成功并添加到了驅動中。上面說的過程是在SDIO設備注冊時的調用流程,mmc_rescan是整個流程主體部分,由它來完成SDIO設備的初始化和添加。其實上面的流程只是創建,初始化,添加SDIO設備的一條線,還有另外的兩條線也會調用mmc_rescan函數進行SDIO設備的上述操作:

(1)??? 加載SDIO host驅動模塊

(2)??? SDIO設備中斷

1.3.2.1????????加載SDIO host驅動模塊

Host作為平臺設備被注冊,前面也有列出相應源碼:

static struct platform_drivermsmsdcc_driver = {

??????? .probe????????? =?msmsdcc_probe,

??????? .suspend??????? = msmsdcc_suspend,

??????? .resume???????? = msmsdcc_resume,

??????? .driver?????? ??= {

??????????????? .name?? = "msm_sdcc",

??????? },

};

?

static int __initmsmsdcc_init(void)

{

????????returnplatform_driver_register(&msmsdcc_driver);

}

?

Probe函數會調用mmc_alloc_host函數(代碼前面已經貼出)來創建mmc_host結構變量,進行必要的初始化之后,調用mmc_add_host函數將它添加到驅動里面:

int mmc_add_host(structmmc_host *host)

{

??????? ……

?

??????? err =device_add(&host->class_dev);

??????? if (err)

??????????????? return err;

????????mmc_start_host(host);

??????? if (!(host->pm_flags &MMC_PM_IGNORE_PM_NOTIFY))

??????????????? register_pm_notifier(&host->pm_notify);

?

??????? return 0;

}

???????Mmc_start_host定義如下:

void mmc_start_host(structmmc_host *host)

{

??????mmc_power_off(host);

?????? mmc_detect_change(host, 0);

}

mmc_power_off中對?ios進行了設置,然后調用?mmc_set_ios(host);

host->ios.power_mode = MMC_POWER_OFF;

?????? host->ios.bus_width = MMC_BUS_WIDTH_1;

?????? host->ios.timing =MMC_TIMING_LEGACY;

?????? mmc_set_ios(host);

mmc_set_ios(host) 中的關鍵語句 host->ops->set_ios(host, ios),實際上在host驅動的probe函數中就已經對host->ops進行了初始化:

……

/*

???????? * Setup MMC host structure

???????? */

????????mmc->ops = &msmsdcc_ops;

……

?

static const structmmc_host_ops msmsdcc_ops = {

??????? .request??????? = msmsdcc_request,

??????? .set_ios??????? =msmsdcc_set_ios,

??????? .enable_sdio_irq =msmsdcc_enable_sdio_irq,

};

?

所以實際上調用的是msmsdcc_set_ios,關于這個函數就不介紹了,可以參考源碼,再看 mmc_detect_change(host, 0),最后一句是:

??????mmc_schedule_delayed_work(&host->detect,delay);

實際上就是調用我們前面說的延時函數 mmc_rescan,后面的流程是一樣的。

1.3.2.2????????SDIO設備中斷

SDIO設備通過SDIO總線與host相連,SDIO總線的DAT[1]即pin8可以作為中斷線使用,當SDIO設備向host產生中斷時,host會對終端做出相應的動作,在host驅動的probe函數中申請并注冊相應的中斷函數:

static int

msmsdcc_probe(structplatform_device *pdev)

{

......

? cmd_irqres = platform_get_resource_byname(pdev, IORESOURCE_IRQ,

?????????????????????????????????????????????????"cmd_irq");

??????? pio_irqres =platform_get_resource_byname(pdev, IORESOURCE_IRQ,

?????????????????????????????????????????????????"pio_irq");

????????stat_irqres =platform_get_resource_byname(pdev, IORESOURCE_IRQ,

??????????????????????????????????????????????????"status_irq");

......

? if (stat_irqres && !(stat_irqres->flags &IORESOURCE_DISABLED)) {

??????????????? unsigned long irqflags =IRQF_SHARED |

??????????????????????? (stat_irqres->flags& IRQF_TRIGGER_MASK);

?

????????????????host->stat_irq = stat_irqres->start;

????????????????ret = request_irq(host->stat_irq,

??????????????????????????????????msmsdcc_platform_status_irq,

?????????????????????????????????irqflags,

?????????????????????????????????DRIVER_NAME " (slot)",

?????????????????????????????????host);

??????????????? if (ret) {

??????????????????????? pr_err("%s: Unableto get slot IRQ %d (%d)\n",

??????????????????????????????mmc_hostname(mmc), host->stat_irq, ret);

??????????????????????? goto clk_disable;

??????????????? }

??????? }

......

}

當產生相應的中斷時調用msmsdcc_platform_status_irq中斷處理函數,這個函數的處理流程:

msmsdcc_platform_status_irq—>

msmsdcc_check_statusà

mmc_detect_changeà

mmc_rescanà

那么,這里為何調用mmc_rescan呢?因為前面說過mmc_rescanrescan函數主要用于SDIO設備的初始化,如果SDIO設備產生中斷不應該是已經初始化可以使用了嗎?其實mmc_rescan還有其它的工作,從函數名就能看出來它還有再掃描檢測功能,即如果設備產生了中斷,mmc_rescan函數一開始就會再次檢測所有掛接在該host上的所有SDIO設備,確認是否存在,如果不存在就做相應的釋放工作,以確保數據的一致性。如果檢測到了新的設備那么它就會創建一個新的mmc_card,初始化并添加該設備。

中斷引發的調用mmc_rescan動作的意義:實現了SDIO設備的熱插拔功能。

1.3.3?????? WIFI驅動流程(二)

?

? 此調用流程由dhd_bus_register發起,通過sdio_register_driver注冊一個sdio設備驅動,然后通過dhdsdio_probe初始化并注冊一個網絡設備,網絡設備的注冊標志著wifi驅動已經成功加載,關于網絡設備的創建,初始化和注冊后面會有詳細介紹,先來理一下上面的調用流程,:

?

dhd_mudule_init—>???????????? //path:dhd/sys/dhd_linux.c

Dhd_bus_registerà??????? // dhd/sys/dhd_sdio.c

Bcmsdh_registerà???????? // bcmsdio/sys/bcmsdh_linux.c

Sdio_function_inità????????????? // bcmsdio/sys/bcmsdh_sdmmc_linux.c

Sdio_register_driverà? // bcmsdio/sys/bcmsdh_sdmmc_linux.c

Bcmsdh_sdmmc_probeà//bcmsdio/sys/bcmsdh_sdmmc_linux.c

Bcmsdh_probeà//bcmsdio/sys/bcmsdh_linux.c

Bcmsdio_probeà //dhd/sys/dhd_sdio.c

這里注意上面兩個紅色標記的函數,sdio_register_driver注冊了一個sdio設備,在匹配成功后調用bcmsdh_sdmmc_probe函數,這個函數會調用bcmsdh_probe。這里有一點要注意:瀏覽bcmsdh_linux.c文件可以看出,在bcmsdh_register函數中,當定義了BCMLXSDMMC宏時,會調用sdio_function_init函數,否則調用driver_register函數:

int

bcmsdh_register(bcmsdh_driver_t*driver)

{

??????? int error = 0;

?

????????drvinfo = *driver; //注意這里,后面會介紹到它的用處

?

#if defined(BCMPLATFORM_BUS)

#if defined(BCMLXSDMMC)

???????SDLX_MSG(("Linux Kernel SDIO/MMC Driver\n"));

????????error =sdio_function_init();

#else

???????SDLX_MSG(("Intel PXA270 SDIO Driver\n"));

????????error =driver_register(&bcmsdh_driver);

#endif /* defined(BCMLXSDMMC) */

??????? return error;

#endif /*defined(BCMPLATFORM_BUS) */

?

#if !defined(BCMPLATFORM_BUS)&& !defined(BCMLXSDMMC)

#if (LINUX_VERSION_CODE <KERNEL_VERSION(2, 6, 0))

??????? if (!(error =pci_module_init(&bcmsdh_pci_driver)))

?? ?????????????return 0;

#else

??????? if (!(error =pci_register_driver(&bcmsdh_pci_driver)))

??????????????? return 0;

#endif

?

??????? SDLX_MSG(("%s: pci_module_initfailed 0x%x\n", __FUNCTION__, error));

#endif /* BCMPLATFORM_BUS */

?

??????? return error;

}

上面的流程中有sdio_function_init的調用出現,所以這里實際上BCMLXSDMMC宏被定義了,bcmsdh_probe函數只是作為一個普通函數被調用,如果不定義該宏,那么bcmsdh_probe函數會被作為驅動匹配設備后第一個調用的函數而被自動調用。

再看看dhdsdio_probe函數調用的玄機,從上面的bcmsdh_register函數可以看出它的參數被傳遞給了drvinfo,看看bcmsdh_register的調用地方:

static bcmsdh_driver_t dhd_sdio = {

????????dhdsdio_probe,

????????dhdsdio_disconnect

};

?

int

dhd_bus_register(void)

{

??????? DHD_TRACE(("%s: Enter\n",__FUNCTION__));

?

??????? return?bcmsdh_register(&dhd_sdio);

}

上面傳遞的參數是dhd_sdio結構變量,被用兩個函數初始化了,那么哪一個是attach呢?需要找到定義bcmsdh_driver_t結構定義的地方:

Path:src/include/bcmsdh.h

/* callback functions */

typedef struct {

??????? /* attach to device */

????????void *(*attach)(uint16 vend_id, uint16 dev_id, uint16 bus,uint16 slot,

???????????????????????uint16 func, uint bustype, void * regsva, osl_t * osh,

???????????????????????void * param);

??????? /* detach from device */

??????? void (*detach)(void *ch);

} bcmsdh_driver_t;

沒錯,就是第一個dhdsdio_probe函數,再來看看什么地方調用了這個attach函數:

Path:bcmsdio/sys/bcmsdh_linux.c

?

#ifndef BCMLXSDMMC

static

#endif /* BCMLXSDMMC */

int bcmsdh_probe(struct device*dev)

{

......

if (!(sdhc->ch =?drvinfo.attach((vendevid>> 16),

????????????????????????????????????????(vendevid & 0xFFFF), 0, 0, 0, 0,

??????????????????????????????????????? (void*)regs, NULL, sdh))) {

???????????? ???SDLX_MSG(("%s: device attachfailed\n", __FUNCTION__));

??????????????? goto err;

??????? }

?

??????? return 0;

......

}

紅色部分的函數調用是drvinfo.attach,就是上面傳遞過來的dhdsdio_probe函數了,仔細閱讀你會發現上面那個bcmsdh_driver_t結構體定義的地方有個說明,即把該結構的成員函數當做callback函數來使用,這就是它的用意所在。

1.3.4?????? 網絡設備注冊流程

?

上面是網絡設備注冊流程,在dhdsdio_probe函數中先后對dhd_attach和dhd_net_attach兩個函數調用,dhd_attach主要用于創建和初始化dhd_info_t和net_device兩個結構變量,然后調用dhd_add_if將創建的net_device變量添加到dhd_info_t變量的iflist列表中(支持多接口)。

Dhd_attach的流程如下:

dhd_pub_t *

dhd_attach(osl_t *osh, structdhd_bus *bus, uint bus_hdrlen)

{

??????? dhd_info_t *dhd = NULL;

??????? struct net_device *net = NULL;

?

......

??????? /* Allocate etherdev, including spacefor private structure */

????????if (!(net = alloc_etherdev(sizeof(dhd)))) {???//網絡設備的創建

??????????????? DHD_ERROR(("%s: OOM -alloc_etherdev\n", __FUNCTION__));

??????????????? goto fail;

??????? }

??????? dhd_state |=DHD_ATTACH_STATE_NET_ALLOC;

?

?

??????? /* Allocate primary dhd_info */

????????if (!(dhd = MALLOC(osh, sizeof(dhd_info_t)))) { //dhd的創建

??????????????? DHD_ERROR(("%s: OOM -alloc dhd_info\n", __FUNCTION__));

??????????????? goto fail;

??????? }

......

/* Set network interface name if it was provided as moduleparameter */

??????? if (iface_name[0]) {

??????????????? int len;

??????????????? char ch;

??????????????? strncpy(net->name,iface_name, IFNAMSIZ);

??????????????? net->name[IFNAMSIZ - 1] = 0;

??????????????? len = strlen(net->name);

??????????????? ch = net->name[len - 1];

??????????????? if ((ch > '9' || ch <'0') && (len < IFNAMSIZ - 2))

??????????????????????? strcat(net->name,"%d");

??????? }

?

????????if (dhd_add_if(dhd, 0, (void *)net, net->name, NULL, 0, 0)== DHD_BAD_IF)???//將前面創建的net添加到iflist列表中

??????????????? goto fail;

??????? dhd_state |= DHD_ATTACH_STATE_ADD_IF;

......

Memcpy(netdev_priv(net), &dhd, sizeof(dhd));?//關聯dhd和net

?

//dhd的初始化工作

}

Dhd_add_if的添加網絡接口流程:

int

dhd_add_if(dhd_info_t *dhd, int ifidx, void *handle, char *name,

????????uint8 *mac_addr,uint32 flags, uint8 bssidx)

{

??????? dhd_if_t *ifp;

?

??????? DHD_TRACE(("%s: idx %d,handle->%p\n", __FUNCTION__, ifidx, handle));

?

??????? ASSERT(dhd && (ifidx <DHD_MAX_IFS));

?

????????ifp =dhd->iflist[ifidx];

??????? if (ifp != NULL) {

??????????????? if (ifp->net != NULL) {

???????????????????????netif_stop_queue(ifp->net);

???????????????????????unregister_netdev(ifp->net);

????????????????????????free_netdev(ifp->net);?? //如果已經存在,釋放net成員

??????????????? }

??????? } else

????????????????if ((ifp = MALLOC(dhd->pub.osh,sizeof(dhd_if_t))) == NULL) {

???????????????????????DHD_ERROR(("%s: OOM - dhd_if_t\n", __FUNCTION__)); ???? //否則,創建一個dhd_if_t結構變量

??????????????????????? return -ENOMEM;

??????????????? }

?

??????? memset(ifp, 0, sizeof(dhd_if_t));

????????ifp->info = dhd;??????//進行系列初始化,添加工作

???????dhd->iflist[ifidx] = ifp;

???????strncpy(ifp->name, name, IFNAMSIZ);

??????? ifp->name[IFNAMSIZ] = '\0';

??????? if (mac_addr != NULL)

????????????????memcpy(&ifp->mac_addr, mac_addr,ETHER_ADDR_LEN);

?

??????? if (handle == NULL) {

??????????????? ifp->state = DHD_IF_ADD;

??????????????? ifp->idx = ifidx;

??????????????? ifp->bssidx = bssidx;

???????????????ASSERT(&dhd->thr_sysioc_ctl.thr_pid >= 0);

???????????????up(&dhd->thr_sysioc_ctl.sema);

??????? } else

????????????????ifp->net = (struct net_device *)handle;?????????????//handle即一個net_device變量

?

??????? return 0;

}

這樣,一個net_device網路設備就被添加到了接口管理列表中了,但是這是網路設備還沒有完成初始化和注冊工作,bcmsdio_probe函數隨后對dhd_net_attach的調用完成了這個操作:

int

dhd_net_attach(dhd_pub_t*dhdp, int ifidx)

{

??????? dhd_info_t *dhd = (dhd_info_t*)dhdp->info;

??????? struct net_device *net = NULL;

??????? int err = 0;

??????? uint8 temp_addr[ETHER_ADDR_LEN] = {0x00, 0x90, 0x4c, 0x11, 0x22, 0x33 };

?

??????? DHD_TRACE(("%s: ifidx %d\n",__FUNCTION__, ifidx));

?

??????? ASSERT(dhd &&dhd->iflist[ifidx]);

?

????????net = dhd->iflist[ifidx]->net;??????????????//首先從剛才添加的接口列表中取出net,然后進行下面的系列初始化工作

??????? ASSERT(net);

//根據內核版本信息,選擇對net成員函數的初始化方式,假設是2.6.30的版本

#if (LINUX_VERSION_CODE <KERNEL_VERSION(2, 6, 31))

??????? ASSERT(!net->open);

????????net->get_stats = dhd_get_stats;

????????net->do_ioctl =dhd_ioctl_entry;

???????net->hard_start_xmit = dhd_start_xmit;

???????net->set_mac_address = dhd_set_mac_address;

???????net->set_multicast_list = dhd_set_multicast_list;

????????net->open =net->stop = NULL;

#else

??????? ASSERT(!net->netdev_ops);

??????? net->netdev_ops = &dhd_ops_virt;

#endif

?

??????? /* Ok, link into the network layer...*/

??????? if (ifidx == 0) {

??????????????? /*

???????????????? * device functions for theprimary interface only

???????????????? */

#if (LINUX_VERSION_CODE <KERNEL_VERSION(2, 6, 31))

????????????????net->open = dhd_open;

???????????????net->stop = dhd_stop;

#else

??????????? ????net->netdev_ops = &dhd_ops_pri;

#endif

??????? } else {

??????????????? /*

???????????????? * We have to use the primaryMAC for virtual interfaces

??????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????3417,1-8????? 66%

???????????????? */

??????????????? memcpy(temp_addr,dhd->iflist[ifidx]->mac_addr, ETHER_ADDR_LEN);

??????????????? /*

???????????????? * Android sets the locallyadministered bit to indicate that this is a

?????????? ??????* portable hotspot.? This will not work in simultaneous AP/STAmode,

???????????????? * nor with P2P.? Need to set the Donlge's MAC address, andthen use that.

???????????????? */

????????????????if(!memcmp(temp_addr, dhd->iflist[0]->mac_addr,

????????????????????????ETHER_ADDR_LEN)) {

??????????????????????? DHD_ERROR(("%sinterface [%s]: set locally administered bit in MAC\n",

??????????????????????? __func__,net->name));

??????????????????????? temp_addr[0] |= 0x02;

??????????????? }

??????? }

?

???????net->hard_header_len = ETH_HLEN + dhd->pub.hdrlen;

#if LINUX_VERSION_CODE >=KERNEL_VERSION(2, 6, 24)

????????net->ethtool_ops = &dhd_ethtool_ops;

#endif /* LINUX_VERSION_CODE>= KERNEL_VERSION(2, 6, 24) */

?

#ifdefined(CONFIG_WIRELESS_EXT)

#if WIRELESS_EXT < 19

????????net->get_wireless_stats = dhd_get_wireless_stats;

#endif /* WIRELESS_EXT < 19*/

#if WIRELESS_EXT > 12

????????net->wireless_handlers = (struct iw_handler_def*)&wl_iw_handler_def;???//這里的初始化工作很重要,之后的ioctl流程會涉及到對它的使用

#endif /* WIRELESS_EXT > 12*/

#endif /*defined(CONFIG_WIRELESS_EXT) */

?

??????? dhd->pub.rxsz =DBUS_RX_BUFFER_SIZE_DHD(net);

????????????????//設置設備地址

????????memcpy(net->dev_addr, temp_addr, ETHER_ADDR_LEN);

?

????????if ((err =register_netdev(net)) != 0) {?????? //注冊net

??????????????? DHD_ERROR(("couldn'tregister the net device, err %d\n", err));

??????????????? goto fail;

??????? }

???????

?

……

}

到這里net網絡設備就被注冊到系統中了,設備準備好了就好對設備進行訪問了

總結

以上是生活随笔為你收集整理的wifi详解(三)的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

999久久国精品免费观看网站 | 国产精品乱码一区二三区 | 天天躁天天操 | 久久黄页 | 午夜久久福利 | 日本巨乳在线 | 久久久久成人精品 | 午夜 久久 tv | 97免费在线观看视频 | 国内精品久久久久久久97牛牛 | 综合在线亚洲 | 日韩欧美在线播放 | 国模吧一区| 麻豆成人在线观看 | 中文在线中文a | 久久66热这里只有精品 | 亚洲精品乱码久久久久 | 999色视频| www.狠狠插.com| 国外av在线 | 亚洲五月激情 | 亚洲精品乱码久久 | 欧美日韩调教 | 久久伦理 | 国产亚洲成人网 | 成人xxxx| 黄色美女免费网站 | 一区二区视频欧美 | 天天操网| 日韩在线观看你懂的 | 日韩69av | 天天操天天干天天玩 | 久久国内精品视频 | 国产亚洲在线观看 | 视频 国产区 | 国产精品麻豆99久久久久久 | 日韩在线视频观看免费 | 激情av在线播放 | 欧美日韩在线视频一区二区 | 久久99国产综合精品免费 | jizz欧美性9 国产一区高清在线观看 | 欧美日韩中文在线观看 | 国产亚洲精品久久久久久久久久久久 | 成人亚洲精品久久久久 | 九九热免费视频在线观看 | 国产精品剧情在线亚洲 | 成人 亚洲 欧美 | av片免费播放 | a级国产片 | 97超碰人人模人人人爽人人爱 | 日韩电影在线视频 | 99精品在线视频观看 | 久草电影免费在线观看 | 91cn国产在线 | 日韩欧美精品在线观看 | 国产精品精品 | 午夜精品电影一区二区在线 | 精品国产乱码久久久久久天美 | 国产精品理论在线观看 | 日韩高清成人在线 | 国产精品一区二区久久 | 国产区第一页 | 免费日韩高清 | 国产日韩欧美精品在线观看 | 国产成人综合精品 | 久久久精品欧美一区二区免费 | 日韩av黄 | 中文资源在线播放 | 久久国产精品视频免费看 | 激情偷乱人伦小说视频在线观看 | 亚洲国产三级在线 | 久草视频观看 | 91精品国产自产91精品 | 91porny九色91啦中文 | 久久精品视频在线看 | 中文字幕在线免费观看 | 精品99在线 | 国产不卡在线看 | 久久婷婷色 | 精品视频在线视频 | 99色网站| 国产无限资源在线观看 | 顶级欧美色妇4khd | 波多野结衣在线播放视频 | 91污在线 | 超碰在线观看av | 国产精品久久久久av | 中文av网| 久久久久久久久毛片精品 | 日韩av三区| 一级成人免费 | 成人全视频免费观看在线看 | 在线免费观看一区二区三区 | 中文在线中文资源 | www.五月婷婷 | 最近日韩免费视频 | 韩国av一区二区 | 97成人精品视频在线观看 | 免费观看一级成人毛片 | 久久综合网色—综合色88 | 国产黄色看片 | 久久综合久久八八 | 久久av免费| 中文字幕日韩伦理 | 久99热| 国产精品女主播一区二区三区 | 婷婷丁香导航 | 日韩精品在线视频免费观看 | 美女视频是黄的免费观看 | 免费黄色在线网站 | 国产亚洲精品久久久网站好莱 | 亚洲最新毛片 | 伊人五月 | 免费av在线网 | 亚洲国产色一区 | 成人毛片100免费观看 | 国产专区精品 | 久久艹欧美 | 女人18片毛片90分钟 | 免费黄色激情视频 | 久久看片 | 精品视频久久久久久 | 国产成人精品一区二区三区网站观看 | 精品国产伦一区二区三区观看说明 | 欧美午夜剧场 | 国产一区福利在线 | 毛片网站在线看 | 激情视频免费在线 | 2019av在线视频 | av一级片 | 国内精自线一二区永久 | 成人黄色视 | 中文字幕国产视频 | 成人在线播放av | 日韩高清一区二区 | 色香蕉在线视频 | 久久久私人影院 | 日韩欧美在线国产 | 日韩69av| 欧美亚洲免费在线一区 | 激情偷乱人伦小说视频在线观看 | 成人免费xxxxxx视频 | 日韩av看片 | 成人黄色小说网 | av三级在线看 | 日韩av影视在线观看 | 五月天综合色激情 | 欧美一二三四在线 | 久草免费在线视频 | 成人免费网站视频 | 亚洲专区欧美专区 | 亚洲欧美国产日韩在线观看 | 99精品毛片 | 亚洲成人xxx | 超碰97免费 | av在线之家电影网站 | 亚洲无吗av | www.888av | 伊人色综合网 | 西西444www高清大胆 | 久久综合网色—综合色88 | 夜夜操天天干, | 日韩国产精品久久 | www.狠狠色.com| 久久黄网站 | 91丝袜美腿| 99国产成+人+综合+亚洲 欧美 | 欧美专区亚洲专区 | 欧美资源 | 欧美天天射 | av高清免费在线 | 91丨精品丨蝌蚪丨白丝jk | 国产va在线| 中文字幕 第二区 | 97超视频免费观看 | 精品久久久久久一区二区里番 | 又黄又爽又色无遮挡免费 | 久久99精品久久久久久久久久久久 | 日日干夜夜操视频 | 91成人网在线观看 | www.狠狠插.com | av网站播放 | 中文字幕一区二区三区乱码在线 | 国产亚州精品视频 | 日韩在线观看av | 久久精品国产精品亚洲精品 | 久久精品最新 | 美女视频久久久 | 亚洲欧洲国产视频 | 久久99久久99免费视频 | 免费午夜在线视频 | 国产高清av | 亚洲视频久久 | 国产亚洲精品久久久久久 | 国产不卡视频在线播放 | 国产精品久久久久久爽爽爽 | 国产中出在线观看 | 国产xx在线| 国产一级黄色免费看 | 国内久久精品 | 国产资源在线免费观看 | 国产精品一区在线观看 | 激情欧美日韩一区二区 | 精品国自产在线观看 | 欧美精品久久久久a | 国产中文字幕在线免费观看 | 国产男女免费完整视频 | 国产一级久久 | 97碰碰视频| 91精品国产福利在线观看 | 手机成人免费视频 | 欧美日韩精品在线一区二区 | 亚洲激情五月 | 久久伊人国产精品 | 亚洲 欧美 综合 在线 精品 | 国产91免费观看 | 天天综合人人 | 韩日精品在线 | 国产高清免费观看 | 欧美激情视频一区 | 天天干天天在线 | 91麻豆国产 | 最新日韩视频 | 国产成人精品a | 麻豆精品传媒视频 | 91成版人在线观看入口 | 久久视频 | av手机在线播放 | 欧美日韩在线免费观看视频 | 就色干综合 | 精品在线不卡 | 国产视频一级 | 国产一区久久久 | 黄a在线| www.国产在线观看 | 国产精品美女网站 | 国产日韩在线看 | 婷婷看片| 国产成人三级一区二区在线观看一 | 亚洲成人动漫在线观看 | 中文字幕免费在线看 | 日本少妇高清做爰视频 | 黄色在线观看免费 | 亚洲国产成人精品在线观看 | 婷婷播播网 | 不卡日韩av | 日韩午夜在线播放 | 99这里有精品 | 五月天欧美精品 | 亚洲黄色免费观看 | 国产理论免费 | 天天干,天天插 | 国产精品久久久久久久久久久不卡 | 国产一区二区免费在线观看 | 色片网站在线观看 | 婷婷在线色 | 久久露脸国产精品 | 麻豆国产精品va在线观看不卡 | 99精品系列 | 97超碰成人| 国产高清视频在线观看 | 中文字幕免费久久 | 亚洲精品乱码久久久久久蜜桃91 | 国产精品高潮呻吟久久av无 | 亚洲激情在线观看 | 天天艹日日干 | 一级黄色电影网站 | 久久久久网站 | 91大神dom调教在线观看 | 制服丝袜一区二区 | 日韩av高清 | 久久五月婷婷综合 | 波多野结衣亚洲一区二区 | 日韩一区二区三区在线观看 | 在线黄色毛片 | 黄色www| 精品在线不卡 | 色偷偷中文字幕 | 探花视频在线观看免费版 | 五月天婷婷免费视频 | 天堂av网站 | 国产黄色成人av | .国产精品成人自产拍在线观看6 | 久久久www成人免费毛片 | 伊人看片 | 免费视频久久久久久久 | 亚洲精品动漫在线 | 麻豆视频免费网站 | 亚洲精品久久久久久国 | 亚洲成成品网站 | 欧美性受极品xxxx喷水 | 久久久久国产成人精品亚洲午夜 | 日韩欧美精品一区二区三区经典 | 日韩电影在线观看中文字幕 | 亚洲最大的av网站 | 91精品啪在线观看国产 | 9992tv成人免费看片 | 在线v片| 久久综合毛片 | 成人精品视频久久久久 | 九九99靖品 | 中文字幕高清有码 | 日韩精品不卡在线 | 国产中文字幕免费 | 日本黄色免费在线 | 国产尤物一区二区三区 | 2019中文字幕网站 | 中文字幕高清在线 | 亚洲天堂网在线播放 | 91视频在线观看大全 | 日韩av网页| 青草草在线视频 | 欧美黑吊大战白妞欧美 | 久久99视频免费观看 | 国产女教师精品久久av | 五月激情丁香婷婷 | 天天操天天射天天插 | 日本最新一区二区三区 | 久久精品一区二区三区国产主播 | 久草在线视频免赞 | 久久成年人视频 | 国产精品12 | 欧美黄网站 | 超碰免费在线公开 | 人人看人人草 | 人人爽夜夜爽 | 中国成人一区 | 日韩r级电影在线观看 | 91麻豆精品国产91久久久更新时间 | 96视频免费在线观看 | 国产精品久久久毛片 | 亚洲人在线 | 国产精品男女视频 | 在线黄色av| 超碰97免费在线 | 五月天综合网 | 中文字幕人成人 | 日韩在线观看一区二区 | 亚洲午夜av | 日本精品久久久久中文字幕 | 免费看av片网站 | 国产美女免费观看 | www.激情五月.com | 91精品啪在线观看国产 | 九月婷婷综合网 | 国产精品美女久久久久久2018 | 人人玩人人添人人 | 日韩理论在线视频 | 91网免费看| 久久精品之 | 亚洲一区 影院 | 久草剧场 | 国产精品中文 | 免费看国产曰批40分钟 | 国产精品成人久久久久久久 | 日韩久久在线 | 激情综合六月 | 热久久最新地址 | 日韩,中文字幕 | 精品久久久一区二区 | 最近中文字幕高清字幕在线视频 | 欧美日韩在线播放 | 91av免费在线观看 | 久久这里只有精品久久 | 国产美女主播精品一区二区三区 | 福利视频一区二区 | 日本激情中文字幕 | 五月天色综合 | 97av视频在线观看 | 久久久久亚洲精品国产 | 亚洲黄色一级电影 | 91香蕉视频黄色 | 午夜国产在线 | 久久综合网色—综合色88 | 日本三级吹潮在线 | 97视频在线免费 | 日韩欧美xxx| 国产亚洲情侣一区二区无 | 999久久国产| 久久成人麻豆午夜电影 | 伊人夜夜| 日韩欧美精品免费 | 中国黄色一级大片 | 国产又粗又猛又黄又爽视频 | 国产网红在线观看 | 久久久久久久久免费视频 | 97在线视 | 久久视频在线观看 | 国产网红在线观看 | 国产精成人品免费观看 | 日韩精品免费一区二区 | 91天堂在线观看 | 国产一区在线观看视频 | 亚洲永久精品在线观看 | 六月丁香婷婷在线 | 色香蕉视频 | 操操日 | av免费观看网站 | 天天操天天玩 | 亚洲色视频| 国产精品网红直播 | 亚洲国内精品在线 | 黄a网站| 亚洲精品日韩一区二区电影 | 久久影视中文字幕 | 草久久久久| 国产女人40精品一区毛片视频 | 久久这里有 | 欧美日韩二区在线 | 国产精品免费大片视频 | 五月天综合 | wwwww.国产 | 国产精品久久视频 | 草久久精品| 成人动漫精品一区二区 | 国产精品久久久久aaaa九色 | 日韩av中文在线观看 | 国产手机在线播放 | av一区在线| 91女人18片女毛片60分钟 | 国产一级片视频 | 欧美一二三区在线观看 | 中文字幕在线播出 | 日韩视频在线不卡 | 日韩在线观看视频网站 | 久久福利国产 | 奇米导航| 天天超碰 | 一区二区三区日韩视频在线观看 | 中文字幕4 | 一区二区影院 | 日本女人逼 | 午夜av日韩| 亚洲精品国产欧美在线观看 | 婷婷丁香久久五月婷婷 | 91精品国产综合久久婷婷香蕉 | 中文字幕在线视频网站 | 久草免费福利在线观看 | 亚洲视频 一区 | 中文在线字幕免 | 国产精品久久久久久电影 | 超碰97在线资源站 | 精品视频在线播放 | 91av视频观看| 午夜私人影院久久久久 | 激情综合亚洲 | 人人舔人人干 | 深爱激情婷婷网 | 久久久久久久久免费视频 | 国产九九热 | 国产在线999 | 97视频播放| 日韩在线观看一区二区 | 免费看国产一级片 | 天天操天天干天天玩 | 黄色大片入口 | v片在线看 | 久久国产电影 | avav99| 日日夜夜天天综合 | 九九九在线观看视频 | 日p在线观看 | 精品国产乱码久久久久久三级人 | 亚洲综合情 | 丁香六月婷婷激情 | 国产精品99久久久久久大便 | 国内综合精品午夜久久资源 | 中文字幕一区在线 | 国产日产av | 人人玩人人添人人澡97 | 国产福利91精品张津瑜 | 国产中文字幕视频在线观看 | 手机在线看片日韩 | 欧美午夜精品久久久久久孕妇 | 中文字幕欧美日韩va免费视频 | 国产精品福利无圣光在线一区 | 欧美日韩国产成人 | 九九热在线视频免费观看 | 久久成人午夜视频 | 色综合久久五月天 | 99精品免费| 亚洲va综合va国产va中文 | 日韩欧美大片免费观看 | 玖玖国产精品视频 | 成人一级片在线观看 | 特级毛片网站 | 日韩精品欧美专区 | 色婷婷啪啪免费在线电影观看 | 免费观看91视频 | 中文字幕一区二区三区在线播放 | 久草在线资源观看 | 人人爱在线视频 | 91福利在线观看 | 狠狠干,狠狠操 | 久久久久久美女 | 欧美激情综合五月色丁香 | 日韩三级一区 | 在线中文字幕播放 | 十八岁免进欧美 | 亚洲激情国产精品 | 成人av免费在线播放 | 日韩精品在线视频 | 91精品视频免费观看 | 永久中文字幕 | 天天干天天弄 | 又黄又爽又色无遮挡免费 | 免费一级特黄录像 | 亚洲日本精品视频 | 亚洲va综合va国产va中文 | 国产在线1区 | 久久精品视频在线观看免费 | 成人永久在线 | 黄色的网站免费看 | 夜夜躁日日躁狠狠久久av | 日日骑| 中文字幕无吗 | 国产精品免费高清 | 成年人黄色在线观看 | 一区二区三区视频 | 亚洲精品资源在线观看 | 91自拍视频在线观看 | 久久久久久久av麻豆果冻 | 麻豆视频免费在线 | 综合中文字幕 | 香蕉视频在线看 | 在线观看视频国产 | 成人在线观看资源 | 日韩高清不卡在线 | 九色在线视频 | 久久中文字幕导航 | 在线中文字幕网站 | 天天操夜夜逼 | 最近中文字幕免费观看 | 欧美大码xxxx | 欧美日韩电影在线播放 | 天天干天天天天 | 韩国av三级 | 亚洲精品视频第一页 | 一二区av | 日韩在线观看第一页 | 人人看看人人 | 精品国产欧美一区二区三区不卡 | 午夜视频播放 | 欧美人体xx | 久久99精品国产一区二区三区 | 中文字幕二区在线观看 | 2024国产精品视频 | 日韩av免费观看网站 | 日韩欧美精品在线视频 | 国产一区二区午夜 | 久久久网页 | 久久综合九色综合久久久精品综合 | 国产欧美综合视频 | 欧美成人按摩 | 成年人国产视频 | 精品国产一区二区三区噜噜噜 | 九九在线视频免费观看 | 亚洲精品视频在线观看视频 | 免费成人在线电影 | 国产精品久久久久婷婷 | 免费视频一区 | 成人一区二区三区在线 | 亚洲人成人99网站 | 日日干干夜夜 | 九九热免费精品视频 | 精品欧美一区二区精品久久 | 成人av动漫在线 | 中文字幕av网站 | 中文字幕视频在线播放 | 久久久久国产成人精品亚洲午夜 | 国产精品一区二区免费 | 97碰在线视频 | 99色视频| 亚洲精品网址在线观看 | 免费观看www小视频的软件 | 久久久久免费精品国产 | 久久国产综合视频 | 97超级碰碰碰碰久久久久 | 91入口在线观看 | 综合色站导航 | 中文在线字幕观看电影 | 亚洲色图激情文学 | 在线视频观看成人 | 国产午夜精品久久久久久久久久 | 亚洲成av人影院 | 天堂av在线中文在线 | 中文字幕在线观看免费高清完整版 | 2017狠狠干 | 九七视频在线观看 | 四虎在线免费视频 | 日韩黄色一级电影 | 欧美黑吊大战白妞欧美 | 人人澡超碰碰97碰碰碰软件 | 一区二区视频在线播放 | 国产视频 亚洲视频 | 免费在线黄色av | 免费在线观看av电影 | 久久精品视频网 | 色播五月激情五月 | 69精品久久| 人人添人人澡人人澡人人人爽 | 国产日韩在线看 | 免费观看av网站 | 国产精品亚洲综合久久 | 亚洲精品午夜一区人人爽 | 亚洲日本黄色 | 日韩精品久久久 | 日本一区二区高清不卡 | 亚洲国产黄色片 | 日本中文字幕在线 | 久久九九久久精品 | 精品一区二区在线免费观看 | 国产123av| 天天要夜夜操 | 免费电影一区二区三区 | 99视频在线精品国自产拍免费观看 | 国产成人在线免费观看 | 久久久片 | 国产亚洲日本 | 久久国产网 | 麻豆国产电影 | 久久综合九色九九 | 欧美一区二区伦理片 | 久草在线免费资源站 | 欧美精品久久久久久久久老牛影院 | 日韩精品无码一区二区三区 | 丁香六月天| 91在线产啪 | 青青色影院| 一区二区三区免费网站 | 韩国精品一区二区三区六区色诱 | 美女天天操 | 久久午夜精品影院一区 | 丝袜美腿亚洲综合 | 一级黄色片在线观看 | 日韩在线高清免费视频 | 成人福利在线播放 | 91黄色小视频| 免费看搞黄视频网站 | 97人人模人人爽人人喊中文字 | 九九热只有这里有精品 | 久久er99热精品一区二区三区 | 免费福利在线视频 | 黄色大片日本免费大片 | 日韩电影在线观看一区二区三区 | 国产精品久久久久久久久久免费 | 99精品久久99久久久久 | 免费在线观看中文字幕 | 国产精品视频全国免费观看 | 国产亚洲精品久久久久久久久久 | 一区二区三区四区在线免费观看 | 99中文在线| 久久99精品国产91久久来源 | 午夜视频一区二区三区 | 成人在线免费视频观看 | 婷婷久久五月 | 在线观看欧美成人 | 69绿帽绿奴3pvideos | 欧美日韩国产色综合一二三四 | 国产视频网站在线观看 | 欧美日本不卡视频 | 国产很黄很色的视频 | 国产成人精品女人久久久 | 久久久久久久久久免费 | 91| 一级做a爱片性色毛片www | 成人黄色免费观看 | 国内偷拍精品视频 | 麻豆果冻剧传媒在线播放 | 91欧美日韩国产 | av动图 | 成人在线观看影院 | 久久99视频精品 | 久久电影国产免费久久电影 | 91视频久久久久 | 亚洲爱视频 | 欧美激情第一区 | 国产高清久久 | 蜜臀一区二区三区精品免费视频 | 日韩免费视频线观看 | 99精品国产一区二区三区不卡 | 午夜.dj高清免费观看视频 | 日韩亚洲在线观看 | 99这里都是精品 | 日日操网站 | 激情影院在线 | 亚洲综合视频在线 | 久草观看视频 | 激情婷婷六月 | 亚洲精品黄色在线观看 | 中文字幕国产视频 | 在线观看中文字幕dvd播放 | 日韩欧美高清免费 | 综合久久精品 | 久久国产精品99国产 | 日韩剧情| 天天操天天干天天摸 | 一本一道久久a久久精品蜜桃 | 在线观看视频一区二区三区 | 精品国产一区二区三区四区在线观看 | 久久久久夜色 | 日韩经典一区二区三区 | 国产一区视频免费在线观看 | 青草草在线 | 韩国精品一区二区三区六区色诱 | 免费在线观看不卡av | 欧美不卡在线 | 免费在线精品视频 | 97免费视频在线播放 | 欧美综合色 | 69绿帽绿奴3pvideos | 天堂av免费 | 九色精品在线 | 国产视频 久久久 | 在线免费三级 | 国产一级片观看 | 免费久久网站 | 久久精品三级 | 成人在线一区二区 | 成人av观看 | 在线免费观看av网站 | www.久久com| 日韩高清无线码2023 | 免费观看黄色av | 欧美在线一二区 | 国产精品久久艹 | 中文字幕一区二区三区四区 | 亚洲精品综合在线 | 97国产一区二区 | 丁香花在线视频观看免费 | 九九热精品视频在线观看 | 日韩激情精品 | 日日操网站 | 最新日本中文字幕 | 99精品乱码国产在线观看 | 免费在线观看成年人视频 | 国产一区免费看 | 国产亚洲一区二区在线观看 | 黄a网| 欧美日韩国产二区三区 | 四虎影视精品永久在线观看 | 国产精品视频线看 | 97超级碰碰 | 黄色毛片大全 | 国产精品久久网站 | a天堂一码二码专区 | 久久99久久久久 | 日韩激情小视频 | 人人爽人人爽人人片av | 丁香视频五月 | 国产精品岛国久久久久久久久红粉 | 久久高清国产 | 中文有码在线视频 | 亚洲成人av在线电影 | 日韩在线观看影院 | av播放在线 | 亚洲一区尤物 | 在线视频观看亚洲 | 久久艹人人| 视频在线观看91 | 在线天堂v | 天天av在线播放 | 中文字幕一区二区在线播放 | 国产99久久精品一区二区300 | 国产小视频在线 | 亚洲一区 影院 | 在线观看视频h | 在线av资源 | 亚洲国产一区二区精品专区 | 激情综合五月天 | 一区二区三区国产精品 | 日韩在线观看视频中文字幕 | 国产精品手机看片 | 欧洲成人免费 | 天堂av在线网站 | 久久久91精品国产一区二区三区 | 韩国精品视频在线观看 | 亚洲欧美在线视频免费 | 成人免费视频网站 | 高清视频一区 | 在线观看免费成人 | 日本精品一二区 | 97成人资源 | 日日碰狠狠躁久久躁综合网 | 国产精品欧美久久 | 国产高清在线免费观看 | 91丝袜美腿| 亚洲成av人影片在线观看 | 欧美精品免费在线 | 亚洲永久精品在线观看 | 亚洲天堂网视频在线观看 | 免费试看一区 | 欧美91视频 | 国产精品久久三 | 日韩和的一区二在线 | 日韩在线影视 | 激情开心站 | 亚洲视频2 | 亚洲五月婷婷 | 国产 成人 久久 | 97精品国产97久久久久久春色 | 国产91在线观看 | 日本一区二区不卡高清 | 日韩一区二区三区在线看 | 视频精品一区二区三区 | 国产区精品在线 | 婷婷午夜天 | a视频免费在线观看 | 国产精品18久久久久久vr | 久久婷婷五月综合色丁香 | 婷婷久久网 | 精品亚洲va在线va天堂资源站 | 欧美a级免费视频 | 亚洲国产精品成人va在线观看 | 激情五月激情综合网 | 欧美国产大片 | www.黄色小说.com | 在线午夜 | 超碰在线97观看 | 干天天 | 日韩日韩日韩日韩 | 操操操夜夜操 | 日本xxxx.com | 国产精品手机在线 | 五月天国产精品 | 日本大片免费观看在线 | 欧美巨乳网 | 久久精品直播 | 久久久精品国产一区二区三区 | 亚洲高清激情 | www.婷婷色| 黄色视屏av | 天天色视频| 国产精品嫩草影视久久久 | 国产精品久久久久一区二区三区共 | 天堂av在线网址 | 免费看片成人 | 久久免费高清视频 | www最近高清中文国语在线观看 | 国产成人精品亚洲 | 久久久国产视频 | 久久久免费看片 | 一本大道久久精品懂色aⅴ 五月婷社区 | 激情久久久久久久久久久久久久久久 | 久久久久久高潮国产精品视 | 亚洲在线视频免费 | 91| 国产视频亚洲视频 | 水蜜桃亚洲一二三四在线 | 国产精品一区二区电影 | 夜色资源站wwwcom | 91亚洲精品乱码久久久久久蜜桃 | 91九色视频网站 | 五月激情姐姐 | 一区二区三区四区五区在线 | 色婷婷啪啪免费在线电影观看 | 亚洲欧美999| 一级a性色生活片久久毛片波多野 | 国产99亚洲| 免费观看黄 | 91久久久久久久一区二区 | 最新av中文字幕 | 有码视频在线观看 | 一区二区三区视频在线 | 日韩性久久 | 国产美女在线免费观看 | 91片黄在线观 | 成年人免费看片网站 | 欧美一级电影在线观看 | 成年人黄色免费看 | 亚洲人成人天堂h久久 | 久久黄色片 | 成人9ⅰ免费影视网站 | 蜜臀av性久久久久av蜜臀三区 | 国产精品久免费的黄网站 | 久久九九国产视频 | 久久黄色a级片 | 久草久| 国产精品视频久久久 | 激情视频免费观看 | 男女日麻批 | 91麻豆.com | 国产另类av | 亚洲美女在线一区 | 久久狠狠亚洲综合 | 人人澡超碰碰 | 亚洲一级电影在线观看 | 香蕉视频网站在线观看 | 久久久精品欧美一区二区免费 | 国产午夜精品av一区二区 | a'aaa级片在线观看 | 91精品视频观看 | 色网站黄 | 亚洲电影黄色 | a黄在线观看 | 江苏妇搡bbbb搡bbbb | 久草国产精品 | 亚洲精品久久久蜜臀下载官网 | 国产 欧美 在线 | 免费又黄又爽的视频 | 国产在线污| 欧美日韩不卡在线 | 久久婷亚洲五月一区天天躁 | 中文字幕在线观看免费高清完整版 | 亚洲 综合 激情 | 91av免费在线观看 | 国产99久久久国产精品 | 欧洲精品久久久久毛片完整版 | 成人国产精品 | 91视频观看免费 | 欧美精品国产综合久久 | 日本69hd | 99精品国产在热久久 | 2019精品手机国产品在线 | 91免费视频网站在线观看 | 91免费看片黄 | 精品国产伦一区二区三区观看说明 | 九九免费精品视频在线观看 | 国产精品成久久久久三级 | 亚洲91网站 | 91久久奴性调教 | 96av在线| 亚洲视屏一区 | 999精品 | 99在线精品视频 | 亚洲欧美经典 | 99在线观看免费视频精品观看 | 六月丁香久久 | 在线观看一区 | 免费看黄色毛片 | 国产一区免费视频 | 免费精品久久久 | 中文字幕免费高清在线 | 91成人在线视频观看 | 国产三级午夜理伦三级 | 色综合激情网 | 日韩av电影手机在线观看 | 国产美女免费观看 | 欧美色就是色 | 日韩视频在线观看免费 | 精品91| 日日夜夜噜噜噜 | 欧美一级片免费 | 色婷婷伊人| 欧美专区日韩专区 | 国产在线观看一 | 免费亚洲黄色 | 婷婷黄色片 | 97干com| 天天爽天天爽天天爽 | 免费在线播放视频 | 99在线精品视频 | 国产精品理论片在线观看 | 成人试看120秒 | 欧美在线视频二区 | 美女视频网 | 99视频精品免费视频 | 免费观看丰满少妇做爰 | 欧美久久久一区二区三区 | 午夜久久久久久久久 | 在线日韩av | 欧洲视频一区 | 天天操天天爽天天干 | 在线观看亚洲a | 久久久久久久福利 | 亚洲狠狠干| 国产清纯在线 | 91激情视频在线 | 在线观看免费av片 | 色欧美88888久久久久久影院 | 国产伦理一区二区 | 美女网站色在线观看 | 成年人在线免费看片 | 色无五月| 国产精品av一区二区 | 五月天丁香综合 | www.91成人 | 免费观看全黄做爰大片国产 | 国产精品一区二区白浆 | 狠狠干夜夜操天天爽 | 久久精品一区二区三区四区 | 97在线看 | www免费黄色 | 91精品视频在线观看免费 | 久久伊人免费视频 | 91精品久久久久久综合乱菊 | 最新精品视频在线 | 日本久久中文 | 免费在线观看a v | 91视频在线免费下载 | 久久久免费看片 | 特黄特色特刺激视频免费播放 | 欧美日韩高清一区二区三区 | www.五月天激情 | 欧美午夜理伦三级在线观看 | 蜜臀久久99精品久久久无需会员 | 国产五十路毛片 | 欧美日韩另类视频 |