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

歡迎訪問 生活随笔!

生活随笔

當(dāng)前位置: 首頁 > 运维知识 > linux >内容正文

linux

linux pcie驱动框架_Linux设备驱动框架设计

發(fā)布時(shí)間:2024/9/19 linux 66 豆豆
生活随笔 收集整理的這篇文章主要介紹了 linux pcie驱动框架_Linux设备驱动框架设计 小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.

引子

Linux操作系統(tǒng)的一大優(yōu)勢(shì)就是支持?jǐn)?shù)以萬計(jì)的芯片設(shè)備,大大小小的芯片廠商工程師都在積極地向Linux kernel提交設(shè)備驅(qū)動(dòng)代碼。能讓這個(gè)目標(biāo)得以實(shí)現(xiàn),這背后隱藏著一個(gè)看不見的技術(shù)優(yōu)勢(shì):Linux內(nèi)核提供了一套易于擴(kuò)展和維護(hù)的設(shè)備驅(qū)動(dòng)框架。Linux內(nèi)核本身提供一套設(shè)備驅(qū)動(dòng)模型,此模型提供了Linux內(nèi)核對(duì)設(shè)備的一般性抽象描述,包括設(shè)備的電源管理、對(duì)象生命周期管理、用戶空間呈現(xiàn)等等。在設(shè)備模型的幫助下,設(shè)備驅(qū)動(dòng)開發(fā)工程師從設(shè)備的一般性抽象中解脫出來。但是每個(gè)設(shè)備的具體功能實(shí)現(xiàn)還需要大量開發(fā)工作,如果每個(gè)設(shè)備都從頭開發(fā),那工作量無疑相當(dāng)巨大。而這些設(shè)備可以按功能進(jìn)行分類,每個(gè)設(shè)備類在業(yè)界或者標(biāo)準(zhǔn)組織中定義了硬件標(biāo)準(zhǔn)規(guī)范,所以針對(duì)每個(gè)設(shè)備類,如果有一個(gè)針對(duì)此設(shè)備類的功能性抽象框架,這將大大加快新設(shè)備的添加和開發(fā)效率。設(shè)備標(biāo)準(zhǔn)規(guī)范的存在,無疑對(duì)設(shè)備驅(qū)動(dòng)框架的設(shè)計(jì)提供了有力的支撐。

但是壞消息也不少:

  • 標(biāo)準(zhǔn)規(guī)范往往落后于產(chǎn)品,特別對(duì)新興設(shè)備來說。
  • 一個(gè)設(shè)備類可能是一套標(biāo)準(zhǔn)規(guī)范定義,也有可能是多套標(biāo)準(zhǔn)規(guī)范定義。
  • 芯片廠商之間,為了達(dá)到提升自己的技術(shù)優(yōu)勢(shì)、市場(chǎng)壁壘等目的,在標(biāo)準(zhǔn)規(guī)范之內(nèi)或之外,做了很多vendor specific的實(shí)現(xiàn)。
  • 這些差異性需求下,一個(gè)設(shè)備類就像一顆樹,其樹干為設(shè)備標(biāo)準(zhǔn)規(guī)范(可能有多個(gè),如下圖),每個(gè)分支為廠商設(shè)備類或設(shè)備子類規(guī)范,而每片樹葉就是每個(gè)具體的設(shè)備。設(shè)備框架的目的就是能幫助驅(qū)動(dòng)工程師簡潔優(yōu)雅地添加一片樹葉。這些差異性需求對(duì)框架設(shè)計(jì)是一個(gè)不小的挑戰(zhàn),如何很好地支持這些需求,是考驗(yàn)優(yōu)秀設(shè)備驅(qū)動(dòng)框架的試金石。

    本文的目的就是總結(jié)一些內(nèi)核設(shè)備驅(qū)動(dòng)框架的優(yōu)秀設(shè)計(jì)方案,以供大家參考。如有疏漏,也歡迎大家留言補(bǔ)充。

    ATA設(shè)備驅(qū)動(dòng)框架設(shè)計(jì)

    現(xiàn)狀描述

    ATA驅(qū)動(dòng)模塊管理眾多的SATA、PATA設(shè)備。以SATA設(shè)備為例,又分支持和不支持 port multiplier功能的,支持port multiplier的。而且SATA總線又存在多種標(biāo)準(zhǔn)規(guī)范定義的(ahci、fsl、 sil24 etc.),就算是用ahci 總線標(biāo)準(zhǔn)的,有些廠商總在某些地方做的跟標(biāo)準(zhǔn)不一致。

    設(shè)計(jì)要點(diǎn)

    • 所有功能點(diǎn)抽象成接口,再抽象成數(shù)據(jù)結(jié)構(gòu)struct ata_port_operations,實(shí)現(xiàn)對(duì)象的多態(tài);
    • 通過struct ata_port_operations的inherits字段實(shí)現(xiàn)對(duì)象的繼承;
    • 新的設(shè)備驅(qū)動(dòng)添加,就是添加新的struct ata_port_operations對(duì)象,而此對(duì)象可以從已有的相似對(duì)象節(jié)點(diǎn)上,通過inherits字段繼承大部分的功能。

    框架設(shè)計(jì)相關(guān)代碼示例:

    struct ata_port_operations {/** Command execution*/int (*qc_defer)(struct ata_queued_cmd *qc);int (*check_atapi_dma)(struct ata_queued_cmd *qc);void (*qc_prep)(struct ata_queued_cmd *qc);unsigned int (*qc_issue)(struct ata_queued_cmd *qc);bool (*qc_fill_rtf)(struct ata_queued_cmd *qc);/** Configuration and exception handling*/int (*cable_detect)(struct ata_port *ap);unsigned long (*mode_filter)(struct ata_device *dev, unsigned long xfer_mask);void (*set_piomode)(struct ata_port *ap, struct ata_device *dev);void (*set_dmamode)(struct ata_port *ap, struct ata_device *dev);int (*set_mode)(struct ata_link *link, struct ata_device **r_failed_dev);unsigned int (*read_id)(struct ata_device *dev, struct ata_taskfile *tf, u16 *id);void (*dev_config)(struct ata_device *dev);void (*freeze)(struct ata_port *ap);void (*thaw)(struct ata_port *ap);ata_prereset_fn_t prereset;ata_reset_fn_t softreset;ata_reset_fn_t hardreset;ata_postreset_fn_t postreset;ata_prereset_fn_t pmp_prereset;ata_reset_fn_t pmp_softreset;ata_reset_fn_t pmp_hardreset;ata_postreset_fn_t pmp_postreset;void (*error_handler)(struct ata_port *ap);void (*lost_interrupt)(struct ata_port *ap);void (*post_internal_cmd)(struct ata_queued_cmd *qc);/** Optional features*/int (*scr_read)(struct ata_link *link, unsigned int sc_reg, u32 *val);int (*scr_write)(struct ata_link *link, unsigned int sc_reg, u32 val);void (*pmp_attach)(struct ata_port *ap);void (*pmp_detach)(struct ata_port *ap);int (*enable_pm)(struct ata_port *ap, enum link_pm policy);void (*disable_pm)(struct ata_port *ap);/** Start, stop, suspend and resume*/int (*port_suspend)(struct ata_port *ap, pm_message_t mesg);int (*port_resume)(struct ata_port *ap);int (*port_start)(struct ata_port *ap);void (*port_stop)(struct ata_port *ap);void (*host_stop)(struct ata_host *host);#ifdef CONFIG_ATA_SFF/** SFF / taskfile oriented ops*/void (*sff_dev_select)(struct ata_port *ap, unsigned int device);u8 (*sff_check_status)(struct ata_port *ap);u8 (*sff_check_altstatus)(struct ata_port *ap);void (*sff_tf_load)(struct ata_port *ap, const struct ata_taskfile *tf);void (*sff_tf_read)(struct ata_port *ap, struct ata_taskfile *tf);void (*sff_exec_command)(struct ata_port *ap,const struct ata_taskfile *tf);unsigned int (*sff_data_xfer)(struct ata_device *dev,unsigned char *buf, unsigned int buflen, int rw);u8 (*sff_irq_on)(struct ata_port *);void (*sff_irq_clear)(struct ata_port *);void (*bmdma_setup)(struct ata_queued_cmd *qc);void (*bmdma_start)(struct ata_queued_cmd *qc);void (*bmdma_stop)(struct ata_queued_cmd *qc);u8 (*bmdma_status)(struct ata_port *ap);void (*drain_fifo)(struct ata_queued_cmd *qc);#endif /* CONFIG_ATA_SFF */ssize_t (*em_show)(struct ata_port *ap, char *buf);ssize_t (*em_store)(struct ata_port *ap, const char *message,size_t size);ssize_t (*sw_activity_show)(struct ata_device *dev, char *buf);ssize_t (*sw_activity_store)(struct ata_device *dev,enum sw_activity val);/** Obsolete*/void (*phy_reset)(struct ata_port *ap);void (*eng_timeout)(struct ata_port *ap);/** ->inherits must be the last field and all the preceding* fields must be pointers.*/const struct ata_port_operations *inherits;[在對(duì)象ata_port_operations 最后一個(gè)字段定義一個(gè)指向ata_port_operations 的指針。ata_port_operations 類似于 C++ 中的 vtable, 這里模仿 C++ 中繼承基類vtable的子類vtable內(nèi)存布局。]};const struct ata_port_operations ata_base_port_ops = {.prereset = ata_std_prereset,.postreset = ata_std_postreset,.error_handler = ata_std_error_handler,};[基類vtable]const struct ata_port_operations sata_port_ops = {.inherits = &ata_base_port_ops,.qc_defer = ata_std_qc_defer,.hardreset = sata_std_hardreset,};[繼承ata_base_port_ops的子類vtable]const struct ata_port_operations sata_pmp_port_ops = {.inherits = &sata_port_ops,.pmp_prereset = ata_std_prereset,.pmp_hardreset = sata_std_hardreset,.pmp_postreset = ata_std_postreset,.error_handler = sata_pmp_error_handler,};static struct ata_port_operations ahci_ops = {.inherits = &sata_pmp_port_ops,.qc_defer = sata_pmp_qc_defer_cmd_switch,.qc_prep = ahci_qc_prep,.qc_issue = ahci_qc_issue,.qc_fill_rtf = ahci_qc_fill_rtf,.freeze = ahci_freeze,.thaw = ahci_thaw,.softreset = ahci_softreset,.hardreset = ahci_hardreset,.postreset = ahci_postreset,.pmp_softreset = ahci_softreset,.error_handler = ahci_error_handler,.post_internal_cmd = ahci_post_internal_cmd,.dev_config = ahci_dev_config,.scr_read = ahci_scr_read,.scr_write = ahci_scr_write,.pmp_attach = ahci_pmp_attach,.pmp_detach = ahci_pmp_detach,.enable_pm = ahci_enable_alpm,.disable_pm = ahci_disable_alpm,.em_show = ahci_led_show,.em_store = ahci_led_store,.sw_activity_show = ahci_activity_show,.sw_activity_store = ahci_activity_store,#ifdef CONFIG_PM.port_suspend = ahci_port_suspend,.port_resume = ahci_port_resume,#endif.port_start = ahci_port_start,.port_stop = ahci_port_stop,};static struct ata_port_operations ahci_vt8251_ops = {.inherits = &ahci_ops,.hardreset = ahci_vt8251_hardreset,};[繼承 ahci_ops的子類 vtable]static const struct ata_port_info ahci_port_info[] = {[board_ahci] ={.flags = AHCI_FLAG_COMMON,.pio_mask = ATA_PIO4,.udma_mask = ATA_UDMA6,.port_ops = &ahci_ops,},[board_ahci_vt8251] ={AHCI_HFLAGS (AHCI_HFLAG_NO_NCQ | AHCI_HFLAG_NO_PMP),.flags = AHCI_FLAG_COMMON,.pio_mask = ATA_PIO4,.udma_mask = ATA_UDMA6,.port_ops = &ahci_vt8251_ops,[初始化 對(duì)象ahci_port_info[board_ahci_vt8251] 的vtable入口port_ops 為ahci_vt8251_ops]},}static void ata_finalize_port_ops(struct ata_port_operations *ops){static DEFINE_SPINLOCK(lock);const struct ata_port_operations *cur;void **begin = (void **)ops;void **end = (void **)&ops->inherits;void **pp;if (!ops || !ops->inherits)return;spin_lock(&lock);for (cur = ops->inherits; cur; cur = cur->inherits) {void **inherit = (void **)cur;for (pp = begin; pp < end; pp++, inherit++)if (!*pp)*pp = *inherit;}for (pp = begin; pp < end; pp++)if (IS_ERR(*pp))*pp = NULL;ops->inherits = NULL;[掃描多重繼承的虛函數(shù)接口]spin_unlock(&lock);}int ata_host_start(struct ata_host *host){int have_stop = 0;void *start_dr = NULL;int i, rc;if (host->flags & ATA_HOST_STARTED)return 0;ata_finalize_port_ops(host->ops);[host對(duì)象初始化時(shí),調(diào)用ata_finalize_port_ops初始化對(duì)象vtable指針host->ops]…}

    PMBus設(shè)備驅(qū)動(dòng)框架設(shè)計(jì)

    現(xiàn)狀描述

    PMBus有一套標(biāo)準(zhǔn)規(guī)范,其中有些是基本功能,有些是可選功能。基本功能是必須要實(shí)現(xiàn)的,而且寄存器接口也進(jìn)行標(biāo)準(zhǔn)化。而可選功能由各設(shè)備廠商自由決定,而且這些可選功能的寄存器接口也無統(tǒng)一規(guī)范,支持PMBus設(shè)備廠商的自定義寄存器。

    設(shè)計(jì)要點(diǎn)

    • 通過struct pmbus_data對(duì)pmbus設(shè)備進(jìn)行抽象性描述,此對(duì)象聚合對(duì)象pmbus_driver_info和pmbus_sensor。通過struct pmbus_driver_info對(duì)pmbus設(shè)備標(biāo)準(zhǔn)規(guī)范進(jìn)行抽象性描述,struct pmbus_sensor對(duì)pmbus傳感器進(jìn)行抽象性描述。
    • 可選功能集通過pmbus_driver_info的u32 func[PMBUS_PAGES]字段描述。
    • 非標(biāo)準(zhǔn)寄存器通過虛擬寄存器(Virtual registers)統(tǒng)一到pmbus驅(qū)動(dòng)框架中。虛擬寄存器到設(shè)備自定義寄存器的映射通過pmbus_driver_info的4個(gè)接口:read_byte_data/read_word_data/write_word_data/write_byte來完成。 
    • 所以當(dāng)添加一個(gè)新設(shè)備時(shí),其差異性需求都封裝在pmbus_driver_info中,這樣pmbus_data和pmbus_sensor做為公共功能則無需修改,
    • 通過構(gòu)造新的pmbus_driver_info對(duì)象即可完成新的設(shè)備驅(qū)動(dòng)的添加。

    框架設(shè)計(jì)相關(guān)代碼示例:

    struct pmbus_data {

    struct device *dev;

    struct device *hwmon_dev;

    u32 flags; /* from platform data */

    int exponent; /* linear mode: exponent for output voltages */

    const struct pmbus_driver_info *info;

    int max_attributes;

    int num_attributes;

    struct attribute_group group;

    struct pmbus_sensor *sensors;

    struct mutex update_lock;

    bool valid;

    unsigned long last_updated; /* in jiffies */

    /*

    * A single status register covers multiple attributes,

    * so we keep them all together.

    */

    u8 status[PB_NUM_STATUS_REG];

    u8 status_register;

    u8 currpage;

    };

    struct pmbus_driver_info {

    int pages; /* Total number of pages */

    enum pmbus_data_format format[PSC_NUM_CLASSES];

    /*

    * Support one set of coefficients for each sensor type

    * Used for chips providing data in direct mode.

    */

    int m[PSC_NUM_CLASSES]; /* mantissa for direct data format */

    int b[PSC_NUM_CLASSES]; /* offset */

    int R[PSC_NUM_CLASSES]; /* exponent */

    u32 func[PMBUS_PAGES]; /* Functionality, per page */

    /*

    * The following functions map manufacturing specific register values

    * to PMBus standard register values. Specify only if mapping is

    * necessary.

    * Functions return the register value (read) or zero (write) if

    * successful. A return value of -ENODATA indicates that there is no

    * manufacturer specific register, but that a standard PMBus register

    * may exist. Any other negative return value indicates that the

    * register does not exist, and that no attempt should be made to read

    * the standard register.

    */

    int (*read_byte_data)(struct i2c_client *client, int page, int reg);

    int (*read_word_data)(struct i2c_client *client, int page, int reg);

    int (*write_word_data)(struct i2c_client *client, int page, int reg,

    u16 word);

    int (*write_byte)(struct i2c_client *client, int page, u8 value);

    /*

    * The identify function determines supported PMBus functionality.

    * This function is only necessary if a chip driver supports multiple

    * chips, and the chip functionality is not pre-determined.

    */

    int (*identify)(struct i2c_client *client,

    struct pmbus_driver_info *info);

    };

    struct pmbus_sensor {

    struct pmbus_sensor *next;

    char name[PMBUS_NAME_SIZE]; /* sysfs sensor name */

    struct device_attribute attribute;

    u8 page; /* page number */

    u16 reg; /* register */

    enum pmbus_sensor_classes class; /* sensor class */

    bool update; /* runtime sensor update needed */

    int data; /* Sensor data.

    Negative if there was a read error */

    };

    static struct pmbus_driver_info tps53667_info = {

    .pages = 1,

    .format[PSC_VOLTAGE_IN] = linear,

    .format[PSC_VOLTAGE_OUT] = vid,

    .format[PSC_TEMPERATURE] = linear,

    .format[PSC_CURRENT_IN] = linear,

    .format[PSC_CURRENT_OUT] = linear,

    .format[PSC_POWER] = linear,

    .read_word_data = tps53667_read_word_data,

    .write_word_data = tps53667_write_word_data,

    .func[0] = PMBUS_HAVE_VIN |

    PMBUS_HAVE_VOUT |

    PMBUS_HAVE_IIN |

    PMBUS_HAVE_IOUT |

    PMBUS_HAVE_PIN |

    PMBUS_HAVE_POUT |

    PMBUS_HAVE_TEMP |

    PMBUS_HAVE_STATUS_VOUT |

    PMBUS_HAVE_STATUS_IOUT |

    PMBUS_HAVE_STATUS_INPUT |

    PMBUS_HAVE_STATUS_TEMP,

    };

    static int tps53667_probe(struct i2c_client *client,

    const struct i2c_device_id *id)

    {

    return pmbus_do_probe(client, id, &tps53667_info);

    }

    int pmbus_do_probe(struct i2c_client *client, const struct i2c_device_id *id,

    struct pmbus_driver_info *info)

    {

    struct device *dev = &client->dev;

    const struct pmbus_platform_data *pdata = dev->platform_data;

    struct pmbus_data *data;

    int ret;

    if (!info)

    return -ENODEV;

    if (!i2c_check_functionality(client->adapter, I2C_FUNC_SMBUS_WRITE_BYTE

    | I2C_FUNC_SMBUS_BYTE_DATA

    | I2C_FUNC_SMBUS_WORD_DATA))

    return -ENODEV;

    data = devm_kzalloc(dev, sizeof(*data), GFP_KERNEL);

    if (!data)

    return -ENOMEM;

    i2c_set_clientdata(client, data);

    mutex_init(&data->update_lock);

    data->dev = dev;

    if (pdata)

    data->flags = pdata->flags;

    data->info = info;

    ret = pmbus_init_common(client, data, info);

    if (ret < 0)

    return ret;

    ret = pmbus_find_attributes(client, data);

    if (ret)

    goto out_kfree;

    /*

    * If there are no attributes, something is wrong.

    * Bail out instead of trying to register nothing.

    */

    if (!data->num_attributes) {

    dev_err(dev, "No attributes foundn");

    ret = -ENODEV;

    goto out_kfree;

    }

    /* Register sysfs hooks */

    ret = sysfs_create_group(&dev->kobj, &data->group);

    if (ret) {

    dev_err(dev, "Failed to create sysfs entriesn");

    goto out_kfree;

    }

    data->hwmon_dev = hwmon_device_register(dev);

    if (IS_ERR(data->hwmon_dev)) {

    ret = PTR_ERR(data->hwmon_dev);

    dev_err(dev, "Failed to register hwmon devicen");

    goto out_hwmon_device_register;

    }

    ret = device_create_file(dev, &dev_attr_clear_fault);

    if (ret)

    goto out_hwmon_device_register;

    return 0;

    out_hwmon_device_register:

    sysfs_remove_group(&dev->kobj, &data->group);

    out_kfree:

    kfree(data->group.attrs);

    return ret;

    }

    USB塊設(shè)備驅(qū)動(dòng)框架設(shè)計(jì)

    現(xiàn)狀描述

    USB塊設(shè)備有一套標(biāo)準(zhǔn)規(guī)范定義,但USB設(shè)備類型眾多(如U盤、MP3播放器、手機(jī)、GPS設(shè)備等等),各廠商水平參差不齊,實(shí)現(xiàn)混亂。而且因?yàn)閁SB規(guī)范相對(duì)較新,一直在演進(jìn)變化中,這也加劇了這一混亂。比如:

    • 設(shè)備描述符中subclass、protocol字段為空或者不正確;
    • 應(yīng)答sense長度非標(biāo)準(zhǔn);
    • INQUIRY設(shè)備請(qǐng)求非標(biāo)準(zhǔn);
    • bulk傳輸協(xié)議中的 tag 不匹配;
    • 類似問題近20個(gè)

    另外,Linux內(nèi)核已有一套通用的SCSI塊設(shè)備驅(qū)動(dòng)框架,如果USB塊設(shè)備能復(fù)用這個(gè)框架,那是一個(gè)最好的選擇。

    設(shè)計(jì)要點(diǎn)

    • 通過struct scsi_host_template對(duì)scsi塊設(shè)備進(jìn)行抽象性描述,USB塊設(shè)備通過創(chuàng)建一個(gè)scsi_host_template對(duì)象,集成到SCSI塊設(shè)備驅(qū)動(dòng)框架中;
    • 通過struct us_data對(duì)USB塊設(shè)備進(jìn)行抽象性描述,通過us_data的fflags字段和unusual_dev字段對(duì)USB塊設(shè)備中的非標(biāo)準(zhǔn)行為進(jìn)行抽象性描述;
    • 通過struct usb_device_id對(duì)USB塊設(shè)備vendor-specific屬性進(jìn)行抽象描述,這樣通過添加一個(gè)新的usb_device_id實(shí)例,即可完成對(duì)一個(gè)新的USB塊設(shè)備驅(qū)動(dòng)的添加。

    框架設(shè)計(jì)相關(guān)代碼示例:

    /* Driver for USB Mass Storage compliant devices

    * SCSI layer glue code

    #define US_DO_ALL_FLAGS

    US_FLAG(SINGLE_LUN, 0x00000001)

    /* allow access to only LUN 0 */

    US_FLAG(NEED_OVERRIDE, 0x00000002)

    /* unusual_devs entry is necessary */

    US_FLAG(SCM_MULT_TARG, 0x00000004)

    /* supports multiple targets */

    US_FLAG(FIX_INQUIRY, 0x00000008)

    /* INQUIRY response needs faking */

    US_FLAG(FIX_CAPACITY, 0x00000010)

    /* READ CAPACITY response too big */

    US_FLAG(IGNORE_RESIDUE, 0x00000020)

    /* reported residue is wrong */

    US_FLAG(BULK32, 0x00000040)

    /* Uses 32-byte CBW length */

    US_FLAG(NOT_LOCKABLE, 0x00000080)

    /* PREVENT/ALLOW not supported */

    US_FLAG(GO_SLOW, 0x00000100)

    /* Need delay after Command phase */

    US_FLAG(NO_WP_DETECT, 0x00000200)

    /* Don't check for write-protect */

    US_FLAG(MAX_SECTORS_64, 0x00000400)

    /* Sets max_sectors to 64 */

    US_FLAG(IGNORE_DEVICE, 0x00000800)

    /* Don't claim device */

    US_FLAG(CAPACITY_HEURISTICS, 0x00001000)

    /* sometimes sizes is too big */

    US_FLAG(MAX_SECTORS_MIN,0x00002000)

    /* Sets max_sectors to arch min */

    US_FLAG(BULK_IGNORE_TAG,0x00004000)

    /* Ignore tag mismatch in bulk operations */

    US_FLAG(SANE_SENSE, 0x00008000)

    /* Sane Sense (> 18 bytes) */

    US_FLAG(CAPACITY_OK, 0x00010000)

    /* READ CAPACITY response is correct */

    US_FLAG(BAD_SENSE, 0x00020000)

    /* Bad Sense (never more than 18 bytes) */

    US_FLAG(NO_READ_DISC_INFO, 0x00040000)

    /* cannot handle READ_DISC_INFO */

    US_FLAG(NO_READ_CAPACITY_16, 0x00080000)

    /* cannot handle READ_CAPACITY_16 */

    US_FLAG(INITIAL_READ10, 0x00100000)

    /* Initial READ(10) (and others) must be retried */

    US_FLAG(WRITE_CACHE, 0x00200000)

    /* Write Cache status is not available */

    #define US_FLAG(name, value) US_FL_##name = value ,

    enum { US_DO_ALL_FLAGS };

    #undef US_FLAG

    struct us_data {

    /* The device we're working with

    * It's important to note:

    * (o) you must hold dev_mutex to change pusb_dev

    */

    struct mutex dev_mutex; /* protect pusb_dev */

    struct usb_device *pusb_dev; /* this usb_device */

    struct usb_interface *pusb_intf; /* this interface */

    struct us_unusual_dev *unusual_dev; /* device-filter entry */

    unsigned long fflags; /* fixed flags from filter */

    unsigned long dflags; /* dynamic atomic bitflags */

    unsigned int send_bulk_pipe; /* cached pipe values */

    unsigned int recv_bulk_pipe;

    unsigned int send_ctrl_pipe;

    unsigned int recv_ctrl_pipe;

    unsigned int recv_intr_pipe;

    /* information about the device */

    char *transport_name;

    char *protocol_name;

    __le32 bcs_signature;

    u8 subclass;

    u8 protocol;

    u8 max_lun;

    u8 ifnum; /* interface number */

    u8 ep_bInterval; /* interrupt interval */

    /* function pointers for this device */

    trans_cmnd transport; /* transport function */

    trans_reset transport_reset; /* transport device reset */

    proto_cmnd proto_handler; /* protocol handler */

    /* SCSI interfaces */

    struct scsi_cmnd *srb; /* current srb */

    unsigned int tag; /* current dCBWTag */

    char scsi_name[32]; /* scsi_host name */

    /* control and bulk communications data */

    struct urb *current_urb; /* USB requests */

    struct usb_ctrlrequest *cr; /* control requests */

    struct usb_sg_request current_sg; /* scatter-gather req. */

    unsigned char *iobuf; /* I/O buffer */

    dma_addr_t iobuf_dma; /* buffer DMA addresses */

    struct task_struct *ctl_thread; /* the control thread */

    /* mutual exclusion and synchronization structures */

    struct completion cmnd_ready; /* to sleep thread on */

    struct completion notify; /* thread begin/end */

    wait_queue_head_t delay_wait; /* wait during reset */

    struct delayed_work scan_dwork; /* for async scanning */

    /* subdriver information */

    void *extra; /* Any extra data */

    extra_data_destructor extra_destructor;/* extra data destructor */

    #ifdef CONFIG_PM

    pm_hook suspend_resume_hook;

    #endif

    /* hacks for READ CAPACITY bug handling */

    int use_last_sector_hacks;

    int last_sector_retries;

    };

    struct usb_device_id {

    /* which fields to match against? */

    __u16 match_flags;

    /* Used for product specific matches; range is inclusive */

    __u16 idVendor;

    __u16 idProduct;

    __u16 bcdDevice_lo;

    __u16 bcdDevice_hi;

    /* Used for device class matches */

    __u8 bDeviceClass;

    __u8 bDeviceSubClass;

    __u8 bDeviceProtocol;

    /* Used for interface class matches */

    __u8 bInterfaceClass;

    __u8 bInterfaceSubClass;

    __u8 bInterfaceProtocol;

    /* Used for vendor-specific interface matches */

    __u8 bInterfaceNumber;

    /* not matched against */

    kernel_ulong_t driver_info

    __attribute__((aligned(sizeof(kernel_ulong_t))));

    };

    /*

    * The table of devices

    */

    #define UNUSUAL_DEV(id_vendor, id_product, bcdDeviceMin, bcdDeviceMax,

    vendorName, productName, useProtocol, useTransport,

    initFunction, flags)

    { USB_DEVICE_VER(id_vendor, id_product, bcdDeviceMin, bcdDeviceMax),

    .driver_info = (flags) }

    struct usb_device_id usb_storage_usb_ids[] = {

    # include "unusual_devs.h"

    { } /* Terminating entry */

    };

    UNUSUAL_DEV( 0x22b8, 0x6426, 0x0101, 0x0101,

    "Motorola",

    "MSnc.",

    USB_SC_DEVICE, USB_PR_DEVICE, NULL,

    US_FL_FIX_INQUIRY | US_FL_FIX_CAPACITY | US_FL_BULK_IGNORE_TAG),

    static int slave_configure(struct scsi_device *sdev)

    {

    ...

    if (us->fflags & US_FL_FIX_CAPACITY)

    sdev->fix_capacity = 1;

    ...

    }

    /*

    * read disk capacity

    */

    static void

    sd_read_capacity(struct scsi_disk *sdkp, unsigned char *buffer)

    {

    ...

    /* Some devices are known to return the total number of blocks,

    * not the highest block number. Some devices have versions

    * which do this and others which do not. Some devices we might

    * suspect of doing this but we don't know for certain.

    *

    * If we know the reported capacity is wrong, decrement it. If

    * we can only guess, then assume the number of blocks is even

    * (usually true but not always) and err on the side of lowering

    * the capacity.

    */

    if (sdp->fix_capacity ||

    (sdp->guess_capacity && (sdkp->capacity & 0x01))) {

    sd_printk(KERN_INFO, sdkp, "Adjusting the sector count "

    "from its reported value: %llun",

    (unsigned long long) sdkp->capacity);

    --sdkp->capacity;

    }

    ...

    }

    struct scsi_host_template usb_stor_host_template = {

    /* basic userland interface stuff */

    .name = "usb-storage",

    .proc_name = "usb-storage",

    .show_info = show_info,

    .write_info = write_info,

    .info = host_info,

    /* command interface -- queued only */

    .queuecommand = queuecommand,

    /* error and abort handlers */

    .eh_abort_handler = command_abort,

    .eh_device_reset_handler = device_reset,

    .eh_bus_reset_handler = bus_reset,

    /* queue commands only, only one command per LUN */

    .can_queue = 1,

    .cmd_per_lun = 1,

    /* unknown initiator id */

    .this_id = -1,

    .slave_alloc = slave_alloc,

    .slave_configure = slave_configure,

    .target_alloc = target_alloc,

    /* lots of sg segments can be handled */

    .sg_tablesize = SCSI_MAX_SG_CHAIN_SEGMENTS,

    /* limit the total size of a transfer to 120 KB */

    .max_sectors = 240,

    /* merge commands... this seems to help performance, but

    * periodically someone should test to see which setting is more

    * optimal.

    */

    .use_clustering = 1,

    /* emulated HBA */

    .emulated = 1,

    /* we do our own delay after a device or bus reset */

    .skip_settle_delay = 1,

    /* sysfs device attributes */

    .sdev_attrs = sysfs_device_attr_list,

    /* module management */

    .module = THIS_MODULE

    };

    嵌入式物聯(lián)網(wǎng)資料分享交流群:707159742 入群有全套學(xué)習(xí)視頻資料電子書免費(fèi)贈(zèng)送!

    資料參考:

    嵌入式底層 - linux設(shè)備驅(qū)動(dòng)之I2C驅(qū)動(dòng)框架 - 創(chuàng)客學(xué)院直播室?www.makeru.com.cn

    總結(jié)

    以上是生活随笔為你收集整理的linux pcie驱动框架_Linux设备驱动框架设计的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網(wǎng)站內(nèi)容還不錯(cuò),歡迎將生活随笔推薦給好友。

    韩日成人av | 久久免费国产电影 | 国产黄色片在线 | 欧美日韩在线观看视频 | 亚洲综合成人在线 | 伊人永久在线 | 日韩免费一区二区在线观看 | 成年人在线观看视频免费 | 中文字幕资源在线观看 | 亚洲国产美女精品久久久久∴ | 在线a人片免费观看视频 | 亚洲视频456| 亚洲免费视频在线观看 | 亚洲 欧洲av | 婷婷成人综合 | www.伊人网 | 国产成年人av | 久久久www免费电影网 | 亚洲情感电影大片 | 久久大片网站 | 国产精品mm| 国产精品久久久久久模特 | 香蕉影院在线播放 | 免费精品视频在线观看 | 91av成人 | 日本中文字幕一二区观 | 这里只有精品视频在线观看 | 少妇精69xxtheporn | 深爱激情开心 | 久久精品99国产精品酒店日本 | 香蕉视频免费看 | 毛片www | 91爱爱免费观看 | 日本公妇在线观看高清 | 视频在线观看亚洲 | 久久久久免费视频 | 精品国产一区二区三区久久久蜜月 | 国产色影院 | 9999国产精品 | 欧美亚洲国产精品久久高清浪潮 | 精品久久久久久国产91 | 久久精品99国产精品日本 | 91一区啪爱嗯打偷拍欧美 | 国产成人精品一区二区三区福利 | 亚洲黄色在线观看 | 久久婷婷一区二区三区 | 久草在线免 | 久久五月情影视 | 狠狠躁夜夜躁人人爽超碰91 | 91亚洲综合| 亚洲精品乱码久久久一二三 | 激情五月婷婷 | 久久久亚洲精华液 | 久久久精品国产一区二区电影四季 | 免费看久久久 | 日韩国产精品一区 | 一级黄色大片在线观看 | 综合亚洲视频 | 亚洲精品tv久久久久久久久久 | jizz999| 成人在线免费小视频 | 天天舔天天射天天操 | 青青草在久久免费久久免费 | 九九热在线观看视频 | www.91国产 | 97超碰人 | 久色免费视频 | 日本中文字幕电影在线免费观看 | 激情综合交 | www.色婷婷 | 天天操综合 | 国产录像在线观看 | 欧美黑人性猛交 | 免费一级片久久 | 在线看黄色的网站 | 日韩免费一级a毛片在线播放一级 | 日韩高清在线不卡 | av电影免费观看 | 99久久99久久免费精品蜜臀 | 国产91精品看黄网站 | 91精品国产成 | 韩日电影在线观看 | 丁香在线观看完整电影视频 | 亚洲日本va午夜在线影院 | avav片| 韩国av免费看 | 日韩欧美国产视频 | 日日干天夜夜 | 91麻豆国产福利在线观看 | 天天综合网~永久入口 | 九草视频在线观看 | 亚洲免费一级电影 | 在线看片中文字幕 | 久草免费看 | 五月婷婷另类国产 | 99日韩精品 | 午夜 久久 tv | 日韩aⅴ视频| 欧美日韩国产伦理 | 天天天天天天干 | 96精品在线| 亚洲精品国产综合99久久夜夜嗨 | 国产精彩视频一区 | 国产a级免费 | 日韩一区正在播放 | 欧美激情视频一二三区 | 免费裸体视频网 | 免费成人结看片 | 国产成人在线观看免费 | 国产麻豆精品95视频 | 婷婷伊人综合 | www.超碰 | 中文字幕a∨在线乱码免费看 | 综合色站导航 | 在线观看日韩视频 | 91精品国产麻豆 | 99久久精品国产观看 | 成人av久久 | 久久这里只有精品首页 | 中文字幕精品一区二区三区电影 | 午夜精品区| 久久国产色 | 久久影视一区 | 亚洲九九九在线观看 | 亚洲精品在线播放视频 | 久久久久电影 | 日韩欧美网址 | 欧美狠狠色 | a视频在线 | av综合av| 在线一区av| 久艹视频在线观看 | 国产99久久精品一区二区300 | 亚洲视频在线免费观看 | 99九九99九九九视频精品 | 丰满少妇在线观看资源站 | 中文字幕 在线 一 二 | 在线观看免费av片 | 97香蕉久久国产在线观看 | 国产精品一区二区久久国产 | 久久精品中文字幕少妇 | 色婷婷成人网 | 成人在线免费观看网站 | 国产精品第一页在线 | 亚洲首页| 天天操天天操天天操天天操天天操 | 国产精品一区二区免费看 | 久久情爱| 欧美精品国产综合久久 | 999国产在线| 国产精品久久 | 精品国产美女 | 久久免费精品 | 久久国产精品影片 | av专区在线| 精品国产网址 | 黄色特级一级片 | 91精品国自产拍天天拍 | 成人在线视频在线观看 | 久久久久久久免费 | 人人精久 | www.狠狠操.com| 国产成人91| 国产免费激情久久 | 深爱激情av | 日韩成人精品在线观看 | 亚洲精品欧洲精品 | 三级动态视频在线观看 | 久久精品久久精品久久 | 麻豆视频国产在线观看 | 成人黄色小说视频 | 午夜久久久久久久久久影院 | 国产小视频在线观看免费 | 国产精品久久久777 成人手机在线视频 | 国产黄色一级片在线 | 国产日韩精品久久 | 狠狠的操 | 色综合人人 | 亚洲在线国产 | 国产精品免费在线播放 | 天天干天天草天天爽 | 国产精品久久久久久久久久免费 | 综合网天天色 | 最近中文字幕免费观看 | 狠狠的干狠狠的操 | 国产成人av网址 | 91麻豆国产 | 精品在线视频一区 | 国产精品久久久久久久免费 | 久久精品一区二区三区国产主播 | 国产精品久久久久久久久大全 | 在线免费观看国产视频 | 手机看片午夜 | 国产黄免费在线观看 | 成人免费ⅴa| 亚洲精品乱码久久 | 欧美日韩性视频 | 国产成人精品一区二区三区 | 免费在线色电影 | 天天插一插 | 婷婷色资源| 激情综合色综合久久 | 亚洲伊人av| aⅴ视频在线 | 五月天激情视频 | 成x99人av在线www | 97超碰人人爱 | 亚洲精品91天天久久人人 | 亚洲欧美日韩中文在线 | 久久久一本精品99久久精品 | 在线99 | 在线观看成人国产 | 久久99久久99免费视频 | 中文字幕在线播放一区 | 国产一区二区三区免费在线 | 色的网站在线观看 | 久久久精品小视频 | 日本性久久 | 91精品国 | 超碰在线9 | 国产在线精品一区二区 | 婷婷综合导航 | 日韩精品在线播放 | 黄色一级片视频 | 天天做天天爱天天爽综合网 | 伊人久久精品久久亚洲一区 | 亚洲黄色免费在线看 | av三级在线播放 | 青草视频在线看 | 久久久久久国产精品久久 | 免费情缘 | 亚洲国产精品久久 | 欧美日韩在线观看视频 | 97超碰成人在线 | 国产一级免费播放 | 欧美日韩一区二区久久 | 亚洲欧美色婷婷 | 国产精品久久嫩一区二区免费 | 免费国产在线观看 | 福利一区二区在线 | 在线精品视频在线观看高清 | 最新日韩视频在线观看 | 亚洲电影黄色 | 干亚洲少妇| 一区二区三区免费播放 | 日韩免费不卡视频 | 91丨九色丨国产在线 | 国产成人精品一区二区三区免费 | 国产三级视频在线 | 欧美成人视| 久久久久久中文字幕 | 国产精品久久人 | 91在线播放综合 | 久久精品国产一区 | 深爱激情五月网 | 久久99热精品这里久久精品 | 毛片网站免费在线观看 | 日日麻批40分钟视频免费观看 | 国产高清黄 | 精品少妇一区二区三区在线 | va视频在线 | 亚洲va韩国va欧美va精四季 | 国产精品毛片一区视频 | 中文字幕免费观看全部电影 | 国产精彩在线视频 | 蜜桃视频色 | 亚洲精品资源 | 岛国av在线 | 不卡av电影在线 | 中文字幕亚洲欧美日韩 | 2021久久 | 黄毛片在线观看 | 午夜精品电影 | 在线免费观看麻豆 | 午夜av一区 | 91在线看视频 | 成 人 黄 色视频免费播放 | www.色就是色| 国产一区免费 | 日b视频国产| 久草99| 中文av字幕在线观看 | 精品久久九九 | av天天澡天天爽天天av | 国产.精品.日韩.另类.中文.在线.播放 | 日韩免费高清在线观看 | 射射色| 在线观看国产中文字幕 | 免费高清看电视网站 | 久久在线精品 | a天堂在线看 | 日韩中文字幕一区 | 免费看黄色小说的网站 | 最近日韩免费视频 | 麻豆成人在线观看 | 国产一区欧美日韩 | 黄色亚洲大片免费在线观看 | 丁香一区二区 | 免费在线观看av的网站 | 婷婷色在线视频 | 日韩高清免费无专码区 | 中文字幕一区二区三区四区久久 | 日本99干网 | 草久在线观看 | 国产专区一 | 视频二区在线 | 色资源在线观看 | 婷婷久久综合九色综合 | 五月色婷 | www.夜夜操| 天天干天天操av | 2019中文字幕网站 | 国产视频精品免费 | 天堂av色婷婷一区二区三区 | 国产一级精品绿帽视频 | 免费视频网 | 韩国av在线播放 | 亚洲精品国产精品国自产 | av免费网站观看 | 不卡视频国产 | 久久人人爽人人爽 | 久久久久久久精 | 黄色精品在线看 | 精品国产一区二区三区噜噜噜 | 俺要去色综合狠狠 | 国产精品夜夜夜一区二区三区尤 | www.天天色.com | 亚洲精品视频在线观看视频 | 日韩字幕在线观看 | 欧美日韩精品免费观看 | 91三级在线观看 | 国产69久久久欧美一级 | 日韩日韩日韩日韩 | 国产成人一区二区啪在线观看 | 国产精品理论片 | 福利视频导航网址 | 欧美经典久久 | 人人精久| 97碰在线| 欧美日韩精品在线视频 | 国产精品午夜av | 免费中午字幕无吗 | 国产成人av福利 | 欧美片一区二区三区 | 激情在线网| 天天碰天天操 | 麻豆网站免费观看 | 黄色免费观看视频 | 九九热.com | 免费观看黄色av | 精品人人人人 | 中文字幕第一页在线vr | 国产亚洲精品久久久久5区 成人h电影在线观看 | 亚洲视频免费 | 综合天堂av久久久久久久 | 中文字幕av免费在线观看 | 一级片观看| 国产只有精品 | 欧美在线一二区 | 国产精品久久久久久久久久久免费 | 色婷婷一 | 成人一区在线观看 | 亚洲 欧美 变态 国产 另类 | 狠狠色免费 | 狠狠地日| 婷婷丁香花| 日韩精品在线视频 | 亚洲成人精品久久久 | 精品国产三级a∨在线欧美 免费一级片在线观看 | 免费观看的av | 日韩,中文字幕 | 丁香网婷婷 | 国产打女人屁股调教97 | 五月天开心| 亚洲精品乱码久久久一二三 | 亚洲国产三级在线观看 | 欧美日韩在线视频一区二区 | 娇妻呻吟一区二区三区 | 五月天亚洲综合 | 在线中文字幕一区二区 | 国产精品一区免费观看 | av线上免费看 | 97香蕉久久国产在线观看 | 亚洲免费视频在线观看 | 免费黄色av. | 成人欧美日韩国产 | 婷婷激情网站 | 91资源在线观看 | 天堂中文在线视频 | 国产自制av | 国产99久久久国产精品免费二区 | 国产精品久久久久久一二三四五 | 欧美日韩在线免费观看视频 | 久久艹人人| 久保带人| 久久综合久久综合这里只有精品 | 久久国产精品第一页 | 日韩视频区| 波多野结衣一区二区三区中文字幕 | 又紧又大又爽精品一区二区 | 国产麻豆果冻传媒在线观看 | av 一区二区三区 | 午夜影院一级 | 少妇bbb好爽 | 日韩成人免费电影 | 99一级片| 久久亚洲欧美 | 久久这里只有精品久久 | 不卡的av在线播放 | 深爱开心激情 | 伊人五月天av | 欧美日韩精品在线观看 | 欧美成年网站 | 国产精品久久久久999 | 五月婷婷播播 | 天天草av | 香蕉在线观看视频 | 国产精品一区二区免费在线观看 | 国产一区二区精品在线 | 国色天香第二季 | 国产精品日韩欧美 | 国产不卡av在线 | 成年人在线观看网站 | 91干干干| 色噜噜在线观看 | 国内精品久久久久影院男同志 | 国产91全国探花系列在线播放 | 国内精品亚洲 | 国产韩国日本高清视频 | 成人蜜桃网 | 日韩在线观看视频中文字幕 | 亚洲专区视频在线观看 | 丁香婷婷自拍 | 91精品啪在线观看国产 | 国产999精品久久久久久麻豆 | 超碰97网站| 狠狠躁夜夜av | 福利一区二区在线 | 99热精品国产| 久久久久久激情 | 丁香六月久久综合狠狠色 | 五月婷婷丁香在线观看 | 久久伊人精品一区二区三区 | 久草在线视频中文 | 成人蜜桃视频 | 欧美色黄 | 日韩在线视频精品 | 四虎永久国产精品 | 免费观看视频黄 | 一区二区在线电影 | 婷婷在线免费 | 国产一二三区在线观看 | 在线免费视 | 久久女同性恋中文字幕 | 国产拍在线| 成人国产精品久久久 | 国产精品破处视频 | 欧美一级专区免费大片 | 麻豆影视网站 | 人人爱在线视频 | 人人狠狠 | 日本三级在线观看中文字 | aaa日本高清在线播放免费观看 | 婷婷视频在线播放 | 中文国产在线观看 | 手机看片1042 | 久久99精品国产91久久来源 | 97激情影院 | 成人亚洲欧美 | 午夜黄网 | 亚洲精选在线 | 国产黄色片一级三级 | 操天天操| 免费国产在线精品 | 免费在线观看日韩 | 特级毛片在线观看 | 六月丁香激情综合 | 中文字幕视频网站 | 国产一级二级在线播放 | 国产伦理一区二区三区 | 亚洲视频aaa | 五月婷婷开心 | 99精品久久精品一区二区 | 日本午夜免费福利视频 | 国产v在线观看 | 在线观看国产一区二区 | 久久久免费播放 | 久久久久久久精 | 亚洲国产精品电影在线观看 | 少妇性aaaaaaaaa视频 | 亚洲精品1234区 | 久久久久久久久国产 | 成人av影院在线观看 | 国产亚洲精品久久久久秋 | 五月婷丁香 | 日日日日干 | 三级a毛片| 午夜视频在线观看一区 | 国产精品高清av | 91亚洲精品久久久蜜桃借种 | 国产999精品 | 成人午夜电影免费在线观看 | 色婷婷综合在线 | 国产黄色特级片 | 成人av资源网站 | 最新真实国产在线视频 | 免费高清在线观看成人 | 99久久这里只有精品 | 成人免费xxx在线观看 | 免费情趣视频 | 久久久首页 | 国产伦理久久精品久久久久_ | 婷婷丁香激情五月 | 九九有精品 | 日韩精品久久一区二区 | 久久九九国产视频 | 一区二区视频在线看 | 日韩午夜剧场 | 日韩久久精品一区二区 | 免费的黄色的网站 | 色婷婷精品| 精品亚洲一区二区三区 | 国产不卡av在线 | 久草精品资源 | 天天干天天干 | 色综合久久综合 | 日韩高清在线一区 | 日韩成人av在线 | 日韩精品免费在线观看视频 | 国产精品九九九九九 | 中文字幕一区二区三区久久 | 久久久午夜精品福利内容 | 911精品视频 | 五月婷网站| 欧美日韩久久 | 少妇bbw搡bbbb搡bbb | 国产成人黄色片 | 亚洲九九九在线观看 | 亚洲精选国产 | 久久99久久99免费视频 | 探花视频免费在线观看 | 午夜精品一二区 | 日本中文字幕网站 | 视频在线日韩 | 99久久超碰中文字幕伊人 | 欧美日韩视频一区二区 | 婷婷丁香六月天 | 麻豆91精品 | 久久超级碰 | 久久黄色网址 | 中文字幕影片免费在线观看 | 国产成人一区二区三区在线观看 | 亚洲天堂视频在线 | 日韩av视屏在线观看 | 国产精品免费久久久久久久久久中文 | 成人欧美在线 | 97精品在线观看 | 九九在线播放 | 亚洲免费在线观看视频 | 精品视频久久 | 干av在线| 精品一区二区在线观看 | 久久不卡国产精品一区二区 | 成年人免费观看在线视频 | 亚洲精品电影在线 | 麻豆影视网 | 久久久999精品视频 国产美女免费观看 | 成人理论在线观看 | 久久久精品 | 99热在线看 | 四虎国产精品免费 | 在线观看免费高清视频大全追剧 | 久久久精品视频网站 | 亚洲情影院 | 中文字幕在线观看视频一区二区三区 | 欧美性天天 | 久久久久 | 最新色站 | 亚洲午夜精品久久久久久久久久久久 | 欧美色操| 亚洲国产中文在线观看 | 在线观看av国产 | 九九九免费视频 | 婷婷开心久久网 | 国产精品毛片完整版 | 精品不卡视频 | 免费网站v| 超碰在线天天 | ww亚洲ww亚在线观看 | 国产精品日韩久久久久 | av一级片在线观看 | 国产日韩在线视频 | 日韩中文字幕在线不卡 | 久久综合福利 | 久久高清毛片 | 日韩av网页 | 最近日本韩国中文字幕 | 成人午夜网 | 99精品视频在线观看免费 | 911国产| 国产一级大片在线观看 | 亚洲精品午夜久久久 | 激情视频一区二区三区 | 中文字幕在线视频一区二区 | 国产91精品一区二区麻豆亚洲 | 四虎永久免费 | 欧美一区日韩精品 | 正在播放国产精品 | 成人91免费视频 | 狠狠色丁香婷婷综合久小说久 | 国产精品高清在线观看 | 91视频在线网址 | 视频一区二区在线 | 欧美激情视频在线免费观看 | 久久草网站 | 久草手机视频 | 色噜噜在线观看 | 国产视频一区在线免费观看 | 免费在线中文字幕 | 国产网站在线免费观看 | 人人爽人人乐 | 亚洲精品在线视频 | 99精品免费在线观看 | 久久a v电影| 中日韩在线视频 | 99精品视频在线观看免费 | 91完整版在线观看 | 亚洲乱码久久久 | 91传媒在线观看 | 亚洲成人av电影 | 91视频麻豆视频 | 国产日韩中文在线 | 国产69精品久久久久99尤 | 韩国一区二区av | 成人午夜电影免费在线观看 | 中文字幕在线观看完整版 | 精品免费观看视频 | 国产一级淫片在线观看 | 亚洲精品一区二区精华 | 久久在线观看视频 | 日韩女同一区二区三区在线观看 | 最新国产在线观看 | 日韩综合视频在线观看 | 久久超级碰视频 | 96av在线| 久久久精品网站 | 91精品久久久久久久99蜜桃 | 久久涩涩网站 | 久久午夜剧场 | 狠狠色婷婷丁香六月 | www.久久久com | 天天综合天天做天天综合 | 国产精品96久久久久久吹潮 | 久久久99国产精品免费 | 久av电影 | 99精品国产在热久久 | 美女黄频网站 | 国产一区二区三区网站 | 国产一区福利在线 | 中文字幕在线播出 | 欧美精品三级 | 97天天干 | 狠狠干中文字幕 | 成人网在线免费视频 | 综合视频在线 | 欧美精品久久久久a | 成人9ⅰ免费影视网站 | 精品免费国产一区二区三区四区 | 日韩视频中文字幕在线观看 | 成人av在线影院 | 国产一级电影在线 | 日韩在线观看视频网站 | 91爱爱电影| 色狠狠操 | 日韩91在线 | 少妇搡bbb| 91在线永久 | 午夜免费久久看 | 国产高清av免费在线观看 | 91亚洲精| 欧美一区二区三区在线视频观看 | 欧美成人按摩 | 国产很黄很色的视频 | 一区二区三区久久精品 | 久久久毛片 | 国产二区精品 | av一级在线 | 69精品视频在线观看 | 日韩久久精品 | 久久久久久久久久久网站 | 91av看片 | 综合婷婷| 人九九精品 | 亚洲精品免费视频 | 日韩成人邪恶影片 | 国产一区二区在线免费 | 国产精品自产拍在线观看桃花 | 国产精品亚洲片夜色在线 | 天天操天天干天天摸 | 久久图 | 日日干视频 | 狠狠色丁香婷婷综合橹88 | 99精品色 | 亚洲第一区在线观看 | 亚洲精品国偷拍自产在线观看蜜桃 | 婷婷久久网站 | 五月天丁香 | 亚洲在线精品视频 | 亚洲欧美综合精品久久成人 | 欧美一级艳片视频免费观看 | 欧美一级艳片视频免费观看 | 天天天天爽| 狠狠操天天干 | 久久精品一二三区 | 成人国产电影在线观看 | 婷婷在线五月 | 国产精品久久久久久69 | 最近高清中文在线字幕在线观看 | 久草网站在线观看 | 最近2019好看的中文字幕免费 | 999亚洲国产996395| 国产午夜精品在线 | 在线成人欧美 | 欧美一级性生活片 | 亚洲全部视频 | 亚洲国产日韩欧美在线 | 日韩毛片在线播放 | 久草香蕉在线 | 国产精品不卡在线播放 | 麻豆传媒视频在线 | 国产福利91精品张津瑜 | 久99热| 日韩久久久久久久 | 欧美一级特黄高清视频 | 精品国产一区二区久久 | 日本精品视频免费 | 亚洲日韩精品欧美一区二区 | 在线观看视频你懂 | 国产欧美精品一区二区三区四区 | 亚洲欧美在线视频免费 | 久久九九免费视频 | 狠狠操狠狠干2017 | 久免费视频| 欧美伦理电影一区二区 | 在线 精品 国产 | 999久久国精品免费观看网站 | 中文字幕在线观看网站 | 成人精品久久 | av在线播放一区二区三区 | 欧美日韩不卡在线 | 最近中文字幕免费av | 免费看一级特黄a大片 | 免费成人在线网站 | 97视频在线免费 | 黄色影院在线播放 | 欧美激情精品久久久久久 | 天天综合网天天 | 射射射av | 色婷婷亚洲精品 | 日韩欧美在线免费 | 日日操网 | 特级毛片爽www免费版 | 91cn国产在线| 91福利试看| 四虎影视8848dvd| 91天天操 | 亚洲国产日韩精品 | 国产精品福利在线播放 | 人人爱人人做人人爽 | www国产亚洲精品久久网站 | 综合中文字幕 | www日韩视频 | 中日韩在线视频 | 色婷婷 亚洲| 亚洲激情在线播放 | 日韩电影在线观看中文字幕 | 午夜久久久久久久久久影院 | 日韩精品字幕 | 在线视频免费观看 | 久久午夜精品影院一区 | 超碰最新网址 | 日韩av在线看 | 天天干,天天草 | 偷拍精品一区二区三区 | www.久热| 不卡日韩av | 九七视频在线观看 | 欧美国产视频在线 | 国产91电影在线观看 | 久久精品精品 | 久艹视频在线观看 | 视频在线观看99 | 在线视频18在线视频4k | 麻豆精品在线 | 欧美日韩视频观看 | 欧美日韩在线免费视频 | 中文av网| 久久免费视频1 | 国产精品免费观看视频 | 精品毛片久久久久久 | 亚洲精品日韩一区二区电影 | 韩日av一区二区 | 久久99精品国产麻豆宅宅 | 开心色婷婷 | 在线中文字幕电影 | 欧美aaa一级 | 在线亚洲午夜片av大片 | 最新日本中文字幕 | 在线观看黄 | 少妇高潮冒白浆 | 91最新网址 | 丁香婷婷激情网 | 久草干| 一区二区精品视频 | 欧美一级片在线免费观看 | 91精品视频观看 | 亚洲精品乱码久久久久久蜜桃欧美 | 97精品国产91久久久久久 | 国产精品久久久久久久久费观看 | 91精彩在线视频 | 色婷五月| 成人黄大片视频在线观看 | 国产精品成人免费一区久久羞羞 | 色瓜 | 久久最新网址 | 日韩av一区二区三区四区 | 高清av免费看 | 色婷在线 | 小草av在线播放 | 狠狠婷婷 | 午夜久久网 | 午夜精品视频福利 | 久久久久久久久久网站 | 日韩电影中文字幕在线 | 伊人射 | 欧洲黄色片 | 91成人免费在线视频 | 91免费观看 | 欧美日韩性视频在线 | 国产生活一级片 | 中文字幕av全部资源www中文字幕在线观看 | 欧美日韩亚洲在线观看 | 亚洲黄色激情小说 | 国产视频每日更新 | 免费看黄色毛片 | 一色屋精品视频在线观看 | 免费在线黄色av | 免费黄色a网站 | 97理论片| 91看片一区二区三区 | 国产91成人 | 亚洲黄色在线观看 | 五月婷婷深开心 | 中文字幕在线不卡国产视频 | 狠狠精品 | 81精品国产乱码久久久久久 | 2023国产精品自产拍在线观看 | 在线观看成人一级片 | av在线小说 | 91麻豆文化传媒在线观看 | 综合网成人 | 中文字幕在线视频国产 | 日韩乱色精品一区二区 | 久久不射电影院 | 国内久久精品 | 欧美色婷| 久久天天躁夜夜躁狠狠85麻豆 | 国产精品地址 | 久久精品专区 | 狠狠操影视 | 成人国产电影在线观看 | 国产 日韩 中文字幕 | 夜夜操网站 | 久久久久久久久久久免费av | 超碰电影在线观看 | 极品美女被弄高潮视频网站 | 欧美日韩1区 | 国产精品久久久久久久久久不蜜月 | 久久一区二区三区超碰国产精品 | 免费看黄色小说的网站 | 中文字幕视频三区 | 久操伊人| 一本一本久久a久久精品综合 | 国产剧情久久 | 99色免费 | 在线电影日韩 | 99精品电影 | 国产精品毛片久久 | 亚洲成av人片一区二区梦乃 | 国产精品免费成人 | 久久艹久久 | 人人艹视频| 欧美日本国产在线观看 | 国产精品久久久久永久免费看 | av黄色免费在线观看 | 欧美精品一区二区在线观看 | 2022久久国产露脸精品国产 | 天天色影院| 日韩大片免费在线观看 | 国产精品福利小视频 | 91成熟丰满女人少妇 | 国产精品刺激对白麻豆99 | 亚洲乱亚洲乱妇 | 日本系列中文字幕 | 国产成人亚洲在线观看 | 久久99久久99久久 | 精品久久久久久久 | 国模视频一区二区 | 91麻豆精品国产午夜天堂 | 久久精品国产99国产 | 九九欧美视频 | 中文字幕一区二 | 992tv在线成人免费观看 | 91精品国产91热久久久做人人 | 亚洲视频中文 | 免费观看国产精品视频 | 亚洲91在线 | 九九99视频 | 日日狠狠 | 免费国产ww | 国产精品久久久久久久久久尿 | av色影院 | 黄色看片 | 日本三级吹潮在线 | 99久久精品国产一区二区成人 | 欧美特一级片 | 91精品在线免费观看视频 | 国产精国产精品 | 悠悠av资源片| 亚洲视频1 | 亚洲精品在线看 | 亚洲国产中文在线 | 日韩欧美一区二区在线观看 | 最新国产在线 | 成年人国产精品 | 国产精品1区2区3区在线观看 | 欧美日韩1区2区 | 久久免费精彩视频 | www.日本色 | 天天躁天天操 | 天天玩天天干天天操 | 亚洲精品乱码久久久久v最新版 | 中文字幕免费观看 | 91系列在线观看 | 国产精品高清在线 | 在线观看成人福利 | 狠狠干综合| 亚洲精品免费在线 | 中文字幕av免费在线观看 | 久久免费成人 | 免费看v片网站 | 成人av电影网址 | 亚洲经典中文字幕 | www99久久 | 久久视频一区二区 | 国产粉嫩在线 | www.91av在线| 欧美日韩视频一区二区三区 | 国产高清在线永久 | 九草视频在线观看 | 欧美少妇的秘密 | 欧美 日韩 性 | 欧美嫩草影院 | 夜夜躁日日躁狠狠躁 | 日韩高清一区 | 成 人 黄 色视频免费播放 | 精品主播网红福利资源观看 | 日韩精品第一区 | 亚洲成av人片在线观看香蕉 | 欧美视频日韩视频 | 粉嫩av一区二区三区免费 | 国产高清绿奴videos | 成人在线视频一区 | 在线99热| 日韩在线视频一区二区三区 | 最近最新中文字幕 | 91最新在线观看 | 欧美日韩亚洲在线观看 | 欧美日韩视频在线一区 | av短片在线观看 | 日韩91av| 免费一级日韩欧美性大片 | 久久久久免费精品 | 久久影视网 | 久久精品99视频 | 亚洲精品中文字幕视频 | 日本在线观看视频一区 | 免费av片在线| 久久午夜国产精品 | 国产精品岛国久久久久久久久红粉 | 亚洲视频中文 | 97成人精品视频在线观看 | www.夜夜骑.com | 日韩av中文字幕在线免费观看 | 成年人在线观看免费视频 | 久久精品区 | www视频在线免费观看 | 久久精品一区二区三区中文字幕 |