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

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 综合教程 >内容正文

综合教程

基于335X的UBOOT网口驱动分析

發布時間:2023/12/29 综合教程 67 生活家
生活随笔 收集整理的這篇文章主要介紹了 基于335X的UBOOT网口驱动分析 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

基于335X的UBOOT網口驅動分析

一、軟硬件平臺資料

1、 開發板:創龍AM3359核心板,網口采用RMII形式

2、 UBOOT版本:U-Boot-2016.05,采用FDT和DM。

參考鏈接:

https://blog.csdn.net/hahachenchen789/article/details/53339181

二、網口相關代碼位置

1、 網口的PINMUX設置

RMII接口的相關PINMUX在MLO中進行設置,具體的設置代碼為
|-board_init_f

|-board_early_init_f

|-set_mux_conf_regs


|-enable_board_pin_mux


configure_module_pin_mux(rmii1_pin_mux);

2、DTS文件中的CPSW的配置

&cpsw_emac0 {
phy_id = <&davinci_mdio>, <0x12>; //phy_id【1】為初始的phy_addr,為SW的PORT2口的ADDR。
phy-mode = "rmii"; //RMII 模式
};

&mac { //未使用此處配置
slaves = <1>;
pinctrl-names = "default", "sleep";
pinctrl-0 = <&cpsw_default>;
pinctrl-1 = <&cpsw_sleep>;
status = "okay";
};

&phy_sel {
rmii-clock-ext; //RMII模式的時鐘為外部時鐘
};

&davinci_mdio { //未使用此處配置
pinctrl-names = "default", "sleep";
pinctrl-0 = <&davinci_mdio_default>;
pinctrl-1 = <&davinci_mdio_sleep>;
status = "okay";
reset-gpios = <&gpio2 5 GPIO_ACTIVE_LOW>;
reset-delay-us = <2>; /* PHY datasheet states 1uS min */
};

3、 網口的初始化設置

網口的初始化在UBOOT中進行,具體設置代碼為

|-board_init_r

|-init_sequence_r

|-initr_net

|- eth_initialize (eth-uclass.c)

三、有關網口的DM&FDT分析

1、 驅動實現方式

此版本的UBOOT中使用了FDT文件進行外設的相關配置,驅動模型使用了DM方式,有關FDT以及DM相關的知識請參考如下文章

https://blog.csdn.net/ooonebook/article/details/53206623

https://blog.csdn.net/ooonebook/article/details/53234020

2、 UBOOT中DM初始化

DM的初始化

.創建根設備root的udevice,存放在gd->dm_root中。

.根設備其實是一個虛擬設備,主要是為uboot的其他設備提供一個掛載點。

.初始化uclass鏈表gd->uclass_root

DM中udevice和uclass的解析

.udevice的創建和uclass的創建

.udevice和uclass的綁定

.uclass_driver和uclass的綁定

.driver和udevice的綁定

.部分driver函數的調用

(1)DM初始化調用過程

dm初始化的接口在dm_init_and_scan中。
可以發現在uboot relocate之前的initf_dm和之后的initr_dm都調用了這個函數。

static int initf_dm(void)

{

#if defined(CONFIG_DM) &&
defined(CONFIG_SYS_MALLOC_F_LEN)


int ret;


ret = dm_init_and_scan(true); // 調用dm_init_and_scan對DM進行初始化和設備的解析


if (ret)


return ret;

#endif


return 0;

}

#ifdef CONFIG_DM

static int initr_dm(void)

{


int ret;


/* Save the pre-reloc driver model and start a new one */


gd->dm_root_f = gd->dm_root; // 存儲relocate之前的根設備


gd->dm_root = NULL;


ret = dm_init_and_scan(false); // 調用dm_init_and_scan對DM進行初始化和設備的解析


if (ret)


return ret;


return 0;

}

#endif

主要區別在于參數。

首先說明一下dts節點中的“u-boot,dm-pre-reloc”屬性,當設置了這個屬性時,則表示這個設備在relocate之前就需要使用。

當dm_init_and_scan的參數為true時,只會對帶有“u-boot,dm-pre-reloc”屬性的節點進行解析。而當參數為false的時候,則會對所有節點都進行解析。

由于“u-boot,dm-pre-reloc”的情況比較少,所以這里只學習參數為false的情況。也就是initr_dm里面的dm_init_and_scan(false);。

dm_init_and_scan(driver/core/root.c)說明

int dm_init_and_scan(bool pre_reloc_only)

{


int ret;


ret = dm_init(); // DM的初始化


if (ret) {


debug("dm_init() failed: %d
", ret);


return ret;

}


ret = dm_scan_platdata(pre_reloc_only); // 從平臺設備中解析udevice和uclass


if (ret) {


debug("dm_scan_platdata() failed: %d
", ret);


return ret;

}


if (CONFIG_IS_ENABLED(OF_CONTROL)) {


ret = dm_scan_fdt(gd->fdt_blob, pre_reloc_only); // 從dtb中解析udevice和uclass


if (ret) {


debug("dm_scan_fdt() failed: %d
", ret);


return ret;


}

}


ret = dm_scan_other(pre_reloc_only);


if (ret)


return ret;


return 0;

}

DM的初始化—dm_init(driver/core/root.c)

#define DM_ROOT_NON_CONST (((gd_t *)gd)->dm_root) // 宏定義根設備指針gd->dm_root

#define DM_UCLASS_ROOT_NON_CONST (((gd_t *)gd)->uclass_root) // 宏定義gd->uclass_root,uclass的鏈表

int dm_init(void)

{


int ret;


if (gd->dm_root) {

// 根設備已經存在,說明DM已經初始化過了


dm_warn("Virtual root driver already exists!
");


return -EINVAL;

}


INIT_LIST_HEAD(&DM_UCLASS_ROOT_NON_CONST);


// 初始化uclass鏈表


ret = device_bind_by_name(NULL, false, &root_info,
&DM_ROOT_NON_CONST);


// DM_ROOT_NON_CONST是指根設備udevice,root_info是表示根設備的設備信息


// device_bind_by_name會查找和設備信息匹配的driver,然后創建對應的udevice和uclass并進行綁定,最后放在DM_ROOT_NON_CONST中。


// device_bind_by_name后續我們會進行說明,這里我們暫時只需要了解root根設備的udevice以及對應的uclass都已經創建完成。

if (ret)


return ret;

#if CONFIG_IS_ENABLED(OF_CONTROL)


DM_ROOT_NON_CONST->of_offset = 0;

#endif


ret = device_probe(DM_ROOT_NON_CONST);


// 對根設備執行probe操作,


// device_probe后續再進行說明


if (ret)


return ret;


return 0;

}

這里就完成的DM的初始化了

1)創建根設備root的udevice,存放在gd->dm_root中。

2)初始化uclass鏈表gd->uclass_root

(2)從平臺設備中解析udevice和uclass—dm_scan_platdata(不涉及)

(3)從dtb中解析udevice和uclass——dm_scan_fdt

對應代碼如下driver/core/root.c

int dm_scan_fdt(const void *blob, bool
pre_reloc_only)

// 此時傳進來的參數blob=gd->fdt_blob,
pre_reloc_only=0

{


return dm_scan_fdt_node(gd->dm_root, blob, 0, pre_reloc_only);

// 直接調用dm_scan_fdt_node

}

int dm_scan_fdt_node(struct udevice
*parent, const void *blob, int offset,


bool pre_reloc_only)

// 此時傳進來的參數

// parent=gd->dm_root,表示以root設備作為父設備開始解析

// blob=gd->fdt_blob,指定了對應的dtb

// offset=0,從偏移0的節點開始掃描

// pre_reloc_only=0,不只是解析relotion之前的設備

{


int ret = 0, err;


/* 以下步驟相當于是遍歷每一個dts節點并且調用lists_bind_fdt對其進行解析 */


for (offset = fdt_first_subnode(blob, offset);


// 獲得blob設備樹的offset偏移下的節點的第一個子節點


offset > 0;


offset = fdt_next_subnode(blob, offset)) {

// 循環查找下一個子節點


if (!fdtdec_get_is_enabled(blob, offset)) {

// 判斷節點狀態是否是disable,如果是的話直接忽略


dm_dbg(" - ignoring
disabled device
");


continue;


}


err = lists_bind_fdt(parent, blob, offset, NULL);

// 解析綁定這個節點,dm_scan_fdt的核心,下面具體分析


if (err && !ret) {


ret = err;


debug("%s: ret=%d
", fdt_get_name(blob, offset, NULL),

ret);


}

}


return ret;

}

lists_bind_fdt是從dtb中解析udevice和uclass的核心。

其具體實現如下:
driver/core/lists.c

int lists_bind_fdt(struct udevice *parent,
const void *blob, int offset,


struct udevice **devp)

// parent指定了父設備,通過blob和offset可以獲得對應的設備的dts節點,對應udevice結構通過devp返回

{


struct driver *driver = ll_entry_start(struct driver, driver);

// 獲取driver table地址


const int n_ents = ll_entry_count(struct driver, driver);

// 獲取driver table長度


const struct udevice_id *id;


struct driver *entry;


struct udevice *dev;


bool found = false;


const char *name;


int result = 0;


int ret = 0;

dm_dbg("bind
node %s
", fdt_get_name(blob, offset, NULL));

// 打印當前解析的節點的名稱


if (devp)


*devp = NULL;


for (entry = driver; entry != driver + n_ents; entry++) {

// 遍歷driver table中的所有driver,具體參考三、4一節


ret = driver_check_compatible(blob, offset, entry->of_match,

&id);

// 判斷driver中的compatibile字段和dts節點是否匹配


name = fdt_get_name(blob, offset, NULL);

// 獲取節點名稱


if (ret == -ENOENT) {


continue;


} else if (ret == -ENODEV) {


dm_dbg("Device '%s' has no compatible string
", name);


break;


} else if (ret) {


dm_warn("Device tree error at offset %d
", offset);


result = ret;


break;


}


dm_dbg(" - found match at
'%s'
", entry->name);


ret = device_bind(parent, entry, name, NULL, offset, &dev);

// 找到對應的driver,調用device_bind進行綁定,會在這個函數中創建對應udevice和uclass并切進行綁定,后面繼續說明


if (ret) {


dm_warn("Error binding driver '%s': %d
", entry->name,

ret);


return ret;


} else {


dev->driver_data = id->data;


found = true;


if (devp)

*devp = dev;

// 將udevice設置到devp指向的地方中,進行返回


}


break;

}

if
(!found && !result && ret != -ENODEV) {


dm_dbg("No match for node '%s'
",

fdt_get_name(blob, offset,
NULL));

}


return result;

}

在device_bind中實現了udevice和uclass的創建和綁定以及一些初始化操作,這里專門學習一下device_bind。

device_bind的實現如下(去除部分代碼)

driver/core/device.c

int device_bind(struct udevice *parent,
const struct driver *drv,


const char *name, void *platdata, int of_offset,


struct udevice **devp)

// parent:父設備

// drv:設備對應的driver

// name:設備名稱

// platdata:設備的平臺數據指針

// of_offset:在dtb中的偏移,即代表了其dts節點

// devp:所創建的udevice的指針,用于返回

{


struct udevice *dev;


struct uclass *uc;


int size, ret = 0;


ret = uclass_get(drv->id, &uc);


// 獲取driver id對應的uclass,如果uclass原先并不存在,那么會在這里創建uclass并其uclass_driver進行綁定


dev = calloc(1, sizeof(struct udevice));


// 分配一個udevice


dev->platdata = platdata; // 設置udevice的平臺數據指針


dev->name = name; // 設置udevice的name


dev->of_offset = of_offset; // 設置udevice的dts節點偏移


dev->parent = parent; // 設置udevice的父設備


dev->driver = drv; // 設置udevice的對應的driver,相當于driver和udevice的綁定


dev->uclass = uc; // 設置udevice的所屬uclass


dev->seq = -1;


dev->req_seq = -1;


if (CONFIG_IS_ENABLED(OF_CONTROL) &&
CONFIG_IS_ENABLED(DM_SEQ_ALIAS)) {


/*


* Some devices, such as a SPI bus, I2C bus and serial ports


* are numbered using aliases.


*


* This is just a 'requested' sequence, and will be


* resolved (and ->seq updated) when the device is probed.


*/


if (uc->uc_drv->flags & DM_UC_FLAG_SEQ_ALIAS) {


if (uc->uc_drv->name && of_offset != -1) {


fdtdec_get_alias_seq(gd->fdt_blob,

uc->uc_drv->name,
of_offset,

&dev->req_seq);


}

// 設置udevice的alias請求序號


}

}


if (!dev->platdata && drv->platdata_auto_alloc_size) {


dev->flags |= DM_FLAG_ALLOC_PDATA;


dev->platdata = calloc(1, drv->platdata_auto_alloc_size);

// 為udevice分配平臺數據的空間,由driver中的platdata_auto_alloc_size決定

}


size = uc->uc_drv->per_device_platdata_auto_alloc_size;


if (size) {


dev->flags |= DM_FLAG_ALLOC_UCLASS_PDATA;


dev->uclass_platdata = calloc(1, size);

// 為udevice分配給其所屬uclass使用的平臺數據的空間,由所屬uclass的driver中的per_device_platdata_auto_alloc_size決定

}


/* put dev into parent's successor list */


if (parent)


list_add_tail(&dev->sibling_node, &parent->child_head);


// 添加到父設備的子設備鏈表中


ret = uclass_bind_device(dev);


// uclass和udevice進行綁定,主要是實現了將udevice鏈接到uclass的設備鏈表中


/* if we fail to bind we remove device from successors and free it */


if (drv->bind) {


ret = drv->bind(dev);


// 執行udevice對應driver的bind函數

}


if (parent && parent->driver->child_post_bind) {


ret = parent->driver->child_post_bind(dev);


// 執行父設備的driver的child_post_bind函數

}


if (uc->uc_drv->post_bind) {


ret = uc->uc_drv->post_bind(dev);


if (ret)


goto fail_uclass_post_bind;


// 執行所屬uclass的post_bind函數

}


if (devp)


*devp = dev;


// 將udevice進行返回


dev->flags |= DM_FLAG_BOUND;


// 設置已經綁定的標志


// 后續可以通過dev->flags & DM_FLAG_ACTIVATED或者device_active宏來判斷設備是否已經被激活


return 0;

在init_sequence_r中的initr_dm中,完成了FDT的解析,解析了所有的外設node,并針對各個節點進行了 udevice和uclass的創建,以及各個組成部分的綁定關系。

注意,這里只是綁定,即調用了driver的bind函數,但是設備還沒有真正激活,也就是還沒有執行設備的probe函數。

將在網口初始化階段進行相關driver的bind操作。

四、網口的初始化過程分析

1、 eth_initialize函數

網口初始化,其中最主要的工作是調用uclass_first_device(UCLASS_ETH,
&dev)函數,從uclass的設備鏈表中獲取第一個udevice,并且進行probe。最終,是通過調用device_probe(dev)進行網口設備的激活和驅動的注冊。下面分析device_probe(dev)的實現的部分過程。

int device_probe(struct udevice *dev)

{


const struct driver *drv;


int size = 0;


int ret;


int seq;


if (dev->flags & DM_FLAG_ACTIVATED)


return 0;

// 表示這個設備已經被激活了


drv = dev->driver;


assert(drv);

// 獲取這個設備對應的driver


/* Allocate private data if requested and not reentered */


if (drv->priv_auto_alloc_size && !dev->priv) {


dev->priv =
alloc_priv(drv->priv_auto_alloc_size, drv->flags);

// 為設備分配私有數據

}


/* Allocate private data if requested and not reentered */


size = dev->uclass->uc_drv->per_device_auto_alloc_size;


if (size && !dev->uclass_priv) {


dev->uclass_priv = calloc(1, size);

// 為設備所屬uclass分配私有數據

}

// 這里過濾父設備的probe


seq = uclass_resolve_seq(dev);


if (seq < 0) {


ret = seq;


goto fail;

}


dev->seq = seq;


dev->flags |= DM_FLAG_ACTIVATED;

// 設置udevice的激活標志

ret = uclass_pre_probe_device(dev);

// uclass在probe device之前的一些函數的調用


if (drv->ofdata_to_platdata && dev->of_offset >= 0) {


ret =
drv->ofdata_to_platdata(dev);

// 調用driver中的ofdata_to_platdata將dts信息轉化為設備的平臺數據

}


if (drv->probe) {

ret = drv->probe(dev);

// 調用driver的probe函數,到這里設備才真正激活了

}


ret = uclass_post_probe_device(dev);


return ret;

}

主要工作歸納如下:

.分配設備的私有數據

.對父設備進行probe

.執行probe device之前uclass需要調用的一些函數

.調用driver的ofdata_to_platdata,將dts信息轉化為設備的平臺數據(重要)

.調用driver的probe函數(重要)

.執行probe device之后uclass需要調用的一些函數

在CPSW.c中有相關定義:

U_BOOT_DRIVER(eth_cpsw) = {

.name = "eth_cpsw",

.id = UCLASS_ETH,

.of_match
= cpsw_eth_ids,

.ofdata_to_platdata
= cpsw_eth_ofdata_to_platdata,

.probe = cpsw_eth_probe,

.ops = &cpsw_eth_ops,

.priv_auto_alloc_size
= sizeof(struct cpsw_priv),

.platdata_auto_alloc_size
= sizeof(struct eth_pdata),

.flags
= DM_FLAG_ALLOC_PRIV_DMA,

};

2、有關DTS配置信息轉化的函數(drv->ofdata_to_platdata)

static int
cpsw_eth_ofdata_to_platdata(struct udevice *dev)

{

struct
eth_pdata *pdata = dev_get_platdata(dev);

struct
cpsw_priv *priv = dev_get_priv(dev);

const
char *phy_mode;

const
char *phy_sel_compat = NULL;

const
void *fdt = gd->fdt_blob;

int
node = dev->of_offset;

int
subnode;

int
slave_index = 0;

int
active_slave;

int
ret;

pdata->iobase
= dev_get_addr(dev);

priv->data.version
= CPSW_CTRL_VERSION_2;

priv->data.bd_ram_ofs
= CPSW_BD_OFFSET;

priv->data.ale_reg_ofs
= CPSW_ALE_OFFSET;

priv->data.cpdma_reg_ofs
= CPSW_CPDMA_OFFSET;

priv->data.mdio_div
= CPSW_MDIO_DIV;

priv->data.host_port_reg_ofs
= CPSW_HOST_PORT_OFFSET,

pdata->phy_interface
= -1;

priv->data.cpsw_base
= pdata->iobase;

priv->data.channels
= fdtdec_get_int(fdt, node, "cpdma_channels", -1);

if
(priv->data.channels <= 0) {

printf("error:
cpdma_channels not found in dt
");

return
-ENOENT;

}

priv->data.slaves
= fdtdec_get_int(fdt, node, "slaves", -1);

if
(priv->data.slaves <= 0) {

printf("error:
slaves not found in dt
");

return
-ENOENT;

}

priv->data.slave_data
= malloc(sizeof(struct cpsw_slave_data) *

priv->data.slaves);

priv->data.ale_entries
= fdtdec_get_int(fdt, node, "ale_entries", -1);

if
(priv->data.ale_entries <= 0) {

printf("error:
ale_entries not found in dt
");

return
-ENOENT;

}

priv->data.bd_ram_ofs
= fdtdec_get_int(fdt, node, "bd_ram_size", -1);

if
(priv->data.bd_ram_ofs <= 0) {

printf("error:
bd_ram_size not found in dt
");

return
-ENOENT;

}

priv->data.mac_control
= fdtdec_get_int(fdt, node, "mac_control", -1);

if
(priv->data.mac_control <= 0) {

printf("error:
ale_entries not found in dt
");

return
-ENOENT;

}

active_slave
= fdtdec_get_int(fdt, node, "active_slave", 0);

priv->data.active_slave
= active_slave;

fdt_for_each_subnode(fdt,
subnode, node) {

int
len;

const
char *name;

name
= fdt_get_name(fdt, subnode, &len);

if
(!strncmp(name, "mdio", 4)) {

u32
mdio_base;

mdio_base
= cpsw_get_addr_by_node(fdt, subnode);

if
(mdio_base == FDT_ADDR_T_NONE) {

error("Not
able to get MDIO address space
");

return
-ENOENT;

}

priv->data.mdio_base
= mdio_base;

}

if
(!strncmp(name, "slave", 5)) {

u32
phy_id[2];

if
(slave_index >= priv->data.slaves)

continue;

phy_mode
= fdt_getprop(fdt, subnode, "phy-mode", NULL);

if
(phy_mode)

priv->data.slave_data[slave_index].phy_if
=

phy_get_interface_by_name(phy_mode);

priv->data.slave_data[slave_index].phy_of_handle
=

fdtdec_lookup_phandle(fdt,
subnode, "phy-handle");

if
(priv->data.slave_data[slave_index].phy_of_handle >= 0) {

priv->data.slave_data[slave_index].phy_addr
=

fdtdec_get_int(gd->fdt_blob,

priv->data.slave_data[slave_index].phy_of_handle,

"reg",
-1);

}
else {

fdtdec_get_int_array(fdt,
subnode, "phy_id", phy_id, 2);

priv->data.slave_data[slave_index].phy_addr
= phy_id[1];

}

slave_index++;

}

if
(!strncmp(name, "cpsw-phy-sel", 12)) {

priv->data.gmii_sel
= cpsw_get_addr_by_node(fdt,

subnode);

if
(priv->data.gmii_sel == FDT_ADDR_T_NONE) {

error("Not
able to get gmii_sel reg address
");

return
-ENOENT;

}

if
(fdt_get_property(fdt, subnode, "rmii-clock-ext",

NULL))

{

priv->data.rmii_clock_external = true;

printf("data.rmii_clock_external is
true
");

}

phy_sel_compat
= fdt_getprop(fdt, subnode, "compatible",

NULL);

if
(!phy_sel_compat) {

printf("Not
able to get gmii_sel compatible
");

return
-ENOENT;

}

}

}

priv->data.slave_data[0].slave_reg_ofs
= CPSW_SLAVE0_OFFSET;

priv->data.slave_data[0].sliver_reg_ofs
= CPSW_SLIVER0_OFFSET;

if
(priv->data.slaves == 2) {

priv->data.slave_data[1].slave_reg_ofs
= CPSW_SLAVE1_OFFSET;

priv->data.slave_data[1].sliver_reg_ofs
= CPSW_SLIVER1_OFFSET;

}

ret
= ti_cm_get_macid(dev, active_slave, pdata->enetaddr);

if
(ret < 0) {

error("cpsw
read efuse mac failed
");

return
ret;

}

pdata->phy_interface
= priv->data.slave_data[active_slave].phy_if;

if
(pdata->phy_interface == -1) {

debug("%s:
Invalid PHY interface '%s'
", __func__, phy_mode);

return
-EINVAL;

}

/*
Select phy interface in control module */

cpsw_phy_sel(priv,
phy_sel_compat, pdata->phy_interface);

return
0;

}

可以看到,在cpsw_eth_ofdata_to_platdata函數中將各種與CPSW有關的平臺數據宏定義以及DTS中的配置信息(包含個子節點)轉化為了平臺數據存儲在了priv->data的相關部分中。主要涉及priv->data的相關設置,此部分重要的信息是MAC的接口形式,比如RMII的設置,RMII時鐘的使能,phy_addr的設置。

3、有關驅動注冊的函數(drv->probe(dev))

static int cpsw_eth_probe(struct udevice *dev)

{

struct cpsw_priv *priv
= dev_get_priv(dev);


printf("cpsw_eth_probe now
");

priv->dev = dev;

return
_cpsw_register(priv);

}

TI對于網卡設備的通用管理是CPSW方式,通過cpsw_priv結構體來進行相關的管理,cpsw_priv結構體中包含有CPSW平臺數據、cpsw_slave的信息、priv->bus(MII接口管理)、phy_device設備的配置及管理等。

cpsw_register(priv)函數主要進行以下工作

(1)、首先是聲明幾個結構體變量,其中包括cpsw的主:cpsw_priv和從:cpsw_slave,然后是設置cpsw的基礎寄存器的地址cpsw_base,然后調用calloc函數為這些結構體分配空間。

struct cpsw_slave *slave;

struct cpsw_platform_data *data =
&priv->data;

void *regs
= (void *)data->cpsw_base;

priv->slaves = malloc(sizeof(struct
cpsw_slave) * data->slaves);

(2)、分配好后對priv結構體中的成員進行初始化,host_port=0表示主機端口號是0,然后成員的寄存器的偏移地址進行初始化。

Priv->host_port =
data->host_port_num;

priv->regs = regs;

priv->host_port_regs = regs + data->host_port_reg_ofs;

priv->dma_regs = regs +
data->cpdma_reg_ofs;

priv->ale_regs = regs + data->ale_reg_ofs;

priv->descs = (void *)regs +
data->bd_ram_ofs;

(3)、對每個salve進行初始化,這里采用for循環的意義在于可能有多個網卡,am335支持雙網卡。

for_each_slave(slave, priv) {

cpsw_slave_setup(slave,
idx, priv);

idx
= idx + 1;

}

(4)、對MDIO接口的操作集進行初始化配置

cpsw_mdio_init(priv->dev->name,
data->mdio_base, data->mdio_div);

.進行了mii_dev設備的創建

.進行了mdio_regs寄存器的配置(set
enable and clock divider)

.進行了cpsw_mdio_read/
cpsw_mdio_write的定義(用此函數對PHY進行讀寫)

.mii_dev設備注冊(加到mii_devs鏈表,并指定為current_mii)

(4)指定priv->bus為上一步創建的設備

priv->bus = miiphy_get_dev_by_name(priv->dev->name);

(5) phydev初始化/配置(重點)

cpsw_phy_init函數定義:

static int
cpsw_phy_init(struct eth_device *dev, struct cpsw_slave *slave)

{

struct cpsw_priv *priv = (struct
cpsw_priv *)dev->priv;

struct phy_device *phydev;

u32 supported = PHY_GBIT_FEATURES;

printf("cpsw_phy_init
");

printf("phy_addr:%d

",slave->data->phy_addr);

phydev = phy_connect(priv->bus,

slave->data->phy_addr,

dev,

slave->data->phy_if);

if (!phydev)

return -1;

phydev->supported &= supported;

phydev->advertising =
phydev->supported;

priv->phydev = phydev;

phy_config(phydev);

return 1;

}

該函數調用phy_connect函數連接網卡,返回的值如果合理就調用phy_config函數對該網卡進行配置,主要是配置網卡的速率和半雙工,自動協商等,此部分需要再進一步調試熟悉。

首先分析phy_connect函數:

struct
phy_device *phy_connect(struct mii_dev *bus, int addr,

struct eth_device *dev,
phy_interface_t interface)

#endif

{

struct phy_device *phydev;

phydev = phy_find_by_mask(bus, 1
<< addr, interface);

if (phydev)

phy_connect_dev(phydev, dev);

else

printf("Could not get
PHY for %s: addr %d
", bus->name, addr);

return phydev;

}

該函數首先調用phy_find_by_mask函數查詢網卡設備,如果存在則調用phy_connect_dev函數連接,否則就打印出錯信息

struct phy_device *phy_find_by_mask(struct mii_dev *bus, unsigned
phy_mask,

phy_interface_t
interface)

{

/* Reset the bus */

if (bus->reset) {

bus->reset(bus);

/* Wait 15ms
to make sure the PHY has come out of hard reset */

udelay(15000);

}

return
get_phy_device_by_mask(bus, phy_mask, interface);

}

該函數主要是調用get_phy_device_by_mask函數進行設備的查找,get_phy_device_by_mask函數的實現至關重要,包含了對于網卡的主要mdio通信。

static struct phy_device
*get_phy_device_by_mask(struct mii_dev *bus,

unsigned
phy_mask, phy_interface_t interface)

{

int
i;

struct
phy_device *phydev;

phydev
= search_for_existing_phy(bus, phy_mask, interface);

if
(phydev)

return
phydev;

/*
Try Standard (ie Clause 22) access */

/*
Otherwise we have to try Clause 45 */

for
(i = 0; i < 5; i++) {

phydev
= create_phy_by_mask(bus, phy_mask,

i
? i : MDIO_DEVAD_NONE, interface);

if
(IS_ERR(phydev))

return
NULL;

if
(phydev)

return
phydev;

}

printf("Phy
%d not found
", ffs(phy_mask) - 1);

return
phy_device_create(bus, ffs(phy_mask) - 1, 0xffffffff, interface);

}

該函數首先調用search_for_existing_phy函數查找當前存在的設備,如果存在則將該設備返回,不存在則調用create_phy_by_mask函數進行創建。重點看下create_phy_by_mask函數

static struct phy_device
*create_phy_by_mask(struct mii_dev *bus,

unsigned
phy_mask, int devad, phy_interface_t interface)

{

u32
phy_id = 0xffffffff;

while
(phy_mask) {

int
addr = ffs(phy_mask) - 1;

int r = get_phy_id(bus, addr, devad,
&phy_id);

/*
If the PHY ID is mostly f's, we didn't find anything */

if
(r == 0 && (phy_id & 0x1fffffff) != 0x1fffffff)

return
phy_device_create(bus, addr, phy_id, interface);

phy_mask
&= ~(1 << addr);

}

return
NULL;

}

該函數調用get_phy_id函數讓處理器通過mdio總線查看網卡寄存器存儲的ID,如果ID都是f,說明沒有ID,就返回空,否則返回phy_device_create函數進行創建一個網卡設備。

get_phy_id函數實現:

int __weak get_phy_id(struct mii_dev *bus,
int addr, int devad, u32 *phy_id)

{

int
phy_reg;

/*
Grab the bits from PHYIR1, and put them

* in the upper half */

phy_reg
= bus->read(bus, addr, devad, MII_PHYSID1);

if
(phy_reg < 0)

return
-EIO;

*phy_id
= (phy_reg & 0xffff) << 16;

/*
Grab the bits from PHYIR2, and put them in the lower half */

phy_reg
= bus->read(bus, addr, devad, MII_PHYSID2);

if
(phy_reg < 0)

return
-EIO;

*phy_id
|= (phy_reg & 0xffff);

return
0;

}

該函數就調用了bus->read總線讀函數,來讀取網卡寄存器的值,這里是讀取寄存器存儲的網卡ID,bus->read函數定義為cpsw_mdio_read

之后的phy_device_create函數為新創建一個phy_device及相關參數,以及相對應的phy_driver。

static struct phy_device
*phy_device_create(struct mii_dev *bus, int addr,

u32 phy_id,

phy_interface_t interface)

{

struct
phy_device *dev;

/*
We allocate the device, and initialize the

* default values */

dev
= malloc(sizeof(*dev));

if
(!dev) {

printf("Failed
to allocate PHY device for %s:%d
",

bus->name,
addr);

return
NULL;

}

memset(dev,
0, sizeof(*dev));

dev->duplex
= -1;

dev->link
= 0;

dev->interface
= interface;

dev->autoneg
= AUTONEG_ENABLE;

dev->addr
= addr;

dev->phy_id
= phy_id;

dev->bus
= bus;

dev->drv
= get_phy_driver(dev, interface);

phy_probe(dev);

bus->phymap[addr]
= dev;

return
dev;

}

其中get_phy_driver會根據phy_id進行phy_driver的查找,若沒有找到,則分配一個"Generic PHY"。

綜上:cpsw_eth_probe的最終結果,是初始化了cpsw_priv各個部分,包括各個參數及mii_dev及phy_dev.

其中phy_dev非常重要,從后面的邏輯看出,phy_dev存在的情況下會根據LINK狀態下的mac_control值對slave->sliver->mac_control寄存器進行配置。這決定了RMII接口的正確配置。

所以,必須有一個phy_dev?或者有一個mac_control值對slave->sliver->mac_control寄存器進行配置?

驅動的初始化及調用

eth_init->eth_get_ops(current)->start(current)來進行網口通信的底層配置

static int cpsw_eth_start(struct udevice
*dev)

{

struct eth_pdata *pdata =
dev_get_platdata(dev);

struct cpsw_priv *priv = dev_get_priv(dev);


printf("cpsw_eth_start_now
");

return _cpsw_init(priv, pdata->enetaddr);

}

static int _cpsw_init(struct cpsw_priv
*priv, u8 *enetaddr)

{

struct
cpsw_slave *slave;

int
i, ret;

printf("_cpsw_init
now
");

/*
soft reset the controller and initialize priv */

setbit_and_wait_for_clear32(&priv->regs->soft_reset);

/*
initialize and reset the address lookup engine */

cpsw_ale_enable(priv,
1);

cpsw_ale_clear(priv,
1);

cpsw_ale_vlan_aware(priv,
0); /* vlan unaware mode */

/*
setup host port priority mapping */

__raw_writel(0x76543210,
&priv->host_port_regs->cpdma_tx_pri_map);

__raw_writel(0,
&priv->host_port_regs->cpdma_rx_chan_map);

/*
disable priority elevation and enable statistics on all ports */

__raw_writel(0,
&priv->regs->ptype);

/*
enable statistics collection only on the host port */

__raw_writel(BIT(priv->host_port),
&priv->regs->stat_port_en);

__raw_writel(0x7,
&priv->regs->stat_port_en);

cpsw_ale_port_state(priv,
priv->host_port, ALE_PORT_STATE_FORWARD);

cpsw_ale_add_ucast(priv,
enetaddr, priv->host_port, ALE_SECURE);

cpsw_ale_add_mcast(priv,
net_bcast_ethaddr, 1 << priv->host_port);

for_active_slave(slave,
priv)

cpsw_slave_init(slave,
priv);

cpsw_update_link(priv);

/*
init descriptor pool */

for
(i = 0; i < NUM_DESCS; i++) {

desc_write(&priv->descs[i],
hw_next,

(i == (NUM_DESCS - 1)) ? 0 :
&priv->descs[i+1]);

}

priv->desc_free
= &priv->descs[0];

/*
initialize channels */

if
(priv->data.version == CPSW_CTRL_VERSION_2) {

memset(&priv->rx_chan,
0, sizeof(struct cpdma_chan));

priv->rx_chan.hdp = priv->dma_regs + CPDMA_RXHDP_VER2;

priv->rx_chan.cp = priv->dma_regs + CPDMA_RXCP_VER2;

priv->rx_chan.rxfree = priv->dma_regs + CPDMA_RXFREE;

memset(&priv->tx_chan,
0, sizeof(struct cpdma_chan));

priv->tx_chan.hdp = priv->dma_regs + CPDMA_TXHDP_VER2;

priv->tx_chan.cp = priv->dma_regs + CPDMA_TXCP_VER2;

}
else {

memset(&priv->rx_chan,
0, sizeof(struct cpdma_chan));

priv->rx_chan.hdp = priv->dma_regs + CPDMA_RXHDP_VER1;

priv->rx_chan.cp = priv->dma_regs + CPDMA_RXCP_VER1;

priv->rx_chan.rxfree = priv->dma_regs + CPDMA_RXFREE;

memset(&priv->tx_chan,
0, sizeof(struct cpdma_chan));

priv->tx_chan.hdp = priv->dma_regs + CPDMA_TXHDP_VER1;

priv->tx_chan.cp = priv->dma_regs + CPDMA_TXCP_VER1;

}

/*
clear dma state */

setbit_and_wait_for_clear32(priv->dma_regs
+ CPDMA_SOFTRESET);

if
(priv->data.version == CPSW_CTRL_VERSION_2) {

for
(i = 0; i < priv->data.channels; i++) {

__raw_writel(0,
priv->dma_regs + CPDMA_RXHDP_VER2 + 4

*
i);

__raw_writel(0,
priv->dma_regs + CPDMA_RXFREE + 4

*
i);

__raw_writel(0,
priv->dma_regs + CPDMA_RXCP_VER2 + 4

*
i);

__raw_writel(0,
priv->dma_regs + CPDMA_TXHDP_VER2 + 4

*
i);

__raw_writel(0,
priv->dma_regs + CPDMA_TXCP_VER2 + 4

* i);

}

}
else {

for
(i = 0; i < priv->data.channels; i++) {

__raw_writel(0,
priv->dma_regs + CPDMA_RXHDP_VER1 + 4

*
i);

__raw_writel(0,
priv->dma_regs + CPDMA_RXFREE + 4

*
i);

__raw_writel(0,
priv->dma_regs + CPDMA_RXCP_VER1 + 4

* i);

__raw_writel(0,
priv->dma_regs + CPDMA_TXHDP_VER1 + 4

*
i);

__raw_writel(0,
priv->dma_regs + CPDMA_TXCP_VER1 + 4

*
i);

}

}

__raw_writel(1,
priv->dma_regs + CPDMA_TXCONTROL);

__raw_writel(1,
priv->dma_regs + CPDMA_RXCONTROL);

/*
submit rx descs */

for
(i = 0; i < PKTBUFSRX; i++) {

ret
= cpdma_submit(priv, &priv->rx_chan, net_rx_packets[i],

PKTSIZE);

if
(ret < 0) {

printf("error
%d submitting rx desc
", ret);

break;

}

}

return
0;

}

其中重點關注下cpsw_update_link(priv)->
cpsw_slave_update_link(slave, priv, &link),這個函數會根據根據現有的priv->phydev設備發起phy_startup(在LINK的狀態下解析phydev->speed、phydev->duplex等狀態),之后根據phy_startupde
的結果更新mac_control,最終通過此函數_raw_writel(mac_control, &slave->sliver->mac_control)將mac_control寫入到相關cpsw_priv
的slave->sliver->mac_control寄存器。只有在link狀態下正確配置了slave->sliver->mac_control寄存器,才能與phy正常進行通信。

以上配置好后,就可以在后續使用PING命令進行測試,PING命令使用之前,還需要配置好IP地址,可使用環境變量進行配置,如setenv ipaddr 192.168.1.30。

總結

以上是生活随笔為你收集整理的基于335X的UBOOT网口驱动分析的全部內容,希望文章能夠幫你解決所遇到的問題。

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

99爱视频在线观看 | 色综合夜色一区 | 99国产一区二区三精品乱码 | 国产专区在线视频 | 精品在线视频一区二区三区 | 免费h视频 | 久久精品视频18 | 在线视频 你懂得 | 在线91播放 | 国产婷婷精品 | 亚洲精品视频在线免费播放 | 福利视频精品 | 国产午夜三级一区二区三桃花影视 | 国产999在线观看 | 国产黄色网| 国产精品久久久久久久久岛 | 亚洲婷婷综合色高清在线 | 精品成人免费 | 4438全国亚洲精品在线观看视频 | 精品综合久久久 | 91看片看淫黄大片 | 97在线免费观看 | 欧美另类一二三四区 | 九九免费在线观看视频 | 9999国产精品 | 欧美孕交vivoestv另类 | 日韩中文在线观看 | 国产高清视频免费最新在线 | 99久久99久久精品国产片果冰 | 伊人国产女 | 国产老太婆免费交性大片 | 亚洲一区二区视频在线播放 | 欧美黑人巨大xxxxx | 天天操天天射天天操 | 天天色天天操综合网 | 狠狠干我 | 一区二区久久 | 天天爽夜夜爽精品视频婷婷 | 在线播放一区二区三区 | 久久这里有精品 | 四虎www| 韩国av免费 | 亚洲日韩中文字幕在线播放 | 成人在线播放免费观看 | 免费一区在线 | 涩涩网站在线播放 | 国产v亚洲v | 91九色最新地址 | 国产999精品视频 | 国产自产高清不卡 | 91污视频在线观看 | 2021国产精品| 日韩成人中文字幕 | 亚洲狠狠婷婷 | 久久久一本精品99久久精品 | 亚洲电影成人 | 国产精品原创视频 | 国产精品久久久久久久久大全 | 欧美精选一区二区三区 | 9在线观看免费高清完整版在线观看明 | 国产色在线视频 | 九九视频网站 | 国产美女精品人人做人人爽 | 精品黄色在线 | 西西4444www大胆视频 | 国产手机在线精品 | 狠狠色丁香久久综合网 | 91热视频| 又黄又刺激视频 | www久草| 国产精品成人自产拍在线观看 | 91桃色视频 | 日韩亚洲欧美中文字幕 | 国产呻吟在线 | 中文字幕第一页在线 | 激情小说网站亚洲综合网 | 久久 在线| 欧美aa在线 | 日韩av电影手机在线观看 | 国产中文字幕视频在线 | 国产亚洲欧美一区 | 日韩欧美视频在线观看免费 | 日本公妇色中文字幕 | 国产精品成人一区二区三区吃奶 | 丁香六月婷婷激情 | 激情自拍av | 日本中文字幕在线电影 | 亚洲国产97在线精品一区 | 成人免费看片网址 | 黄色毛片视频 | 国产精品麻豆91 | 999视频在线观看 | av网站在线观看免费 | 免费欧美 | 欧洲av在线 | 免费韩国av| 日日夜夜人人精品 | 中文字幕国产视频 | 亚洲精品视频在线免费播放 | 国产亚洲精品福利 | 91网免费看 | 亚洲最新合集 | 国产主播大尺度精品福利免费 | 免费看片网站91 | 日韩视频二区 | 尤物九九久久国产精品的分类 | av看片网 | 日韩二区在线观看 | 国产区在线看 | 国产一级淫片在线观看 | 日本精品视频在线观看 | 97人人模人人爽人人少妇 | 狠狠干夜夜操天天爽 | 超碰免费在线公开 | 一区二区激情视频 | 91视频免费 | 六月丁香激情网 | 国产理论在线 | 成人av在线资源 | 日本在线观看一区二区三区 | 国产精品久久免费看 | 久久精品欧美日韩精品 | 日本天天色 | 91久久久久久久一区二区 | 国产精品一区二区三区在线看 | 久草在线视频新 | 国产精品11| 天天要夜夜操 | 亚洲精品乱码久久久久久蜜桃91 | 一区二区三区动漫 | 能在线观看的日韩av | 亚洲综合欧美精品电影 | 五月激情婷婷丁香 | 亚洲三级在线 | 国产一区二区在线免费 | 六月丁香综合 | 久久久夜色 | 色资源中文字幕 | 国产99久久久国产精品免费看 | 狠狠狠的干 | 日韩免费在线看 | 精品在线观看免费 | 精品国产成人在线 | 综合网天天 | av免费在线观看网站 | 亚洲国产精品va在线看黑人动漫 | 成人av影视观看 | 日韩av影视在线观看 | 日韩美女黄色片 | 久草精品视频在线观看 | 国产亚洲视频系列 | 日韩免费视频线观看 | 亚洲综合视频在线播放 | 五月天婷婷狠狠 | 久久久香蕉视频 | 中文字幕在线观看的网站 | 天天综合操 | 一区二区视频欧美 | 中文字幕视频一区二区 | 亚洲精品免费在线 | 国产又粗又猛又黄 | 亚洲精品高清视频在线观看 | 日日夜夜狠狠干 | 免费观看91 | 日韩理论在线视频 | 久久久综合色 | 精品国产一区二区三区不卡 | 国产精品久久一区二区三区, | 99久久超碰中文字幕伊人 | 九九免费精品视频 | www亚洲精品 | 超碰在线色| 久久久999免费视频 日韩网站在线 | 六月色 | 久久电影国产免费久久电影 | 在线国产专区 | 一区二区免费不卡在线 | 成人黄色大片网站 | 日韩免费视频网站 | 欧美精品一区二区三区一线天视频 | 国产成人在线免费观看 | 久久天堂网站 | 国内成人综合 | 久久精品亚洲 | 狠狠干狠狠久久 | 最近中文字幕国语免费av | 超碰在线网 | 九九三级毛片 | 在线观看午夜av | 国产精品久久久久久久久久久久久久 | 狠狠的日日 | 在线观看成人国产 | 国产成人精品一区二区三区免费 | 在线免费观看不卡av | 婷婷伊人五月天 | 一区在线播放 | 美女久久久 | 免费亚洲精品视频 | 国产色黄网站 | 香蕉视频在线免费 | 射射色 | 欧美精品v国产精品v日韩精品 | 三级动图 | 天天射色综合 | 婷婷色中文网 | 在线观看岛国av | 国产精品久久久久久久免费观看 | 国产一区视频免费在线观看 | 国内成人精品2018免费看 | 国产免费一区二区三区网站免费 | 久久午夜羞羞影院 | 91福利在线导航 | 欧美专区日韩专区 | 综合精品久久 | 亚洲第一av在线播放 | 亚洲在线观看av | 国精产品一二三线999 | 日韩一区二区三区免费视频 | 日韩高清成人 | 日本99干网 | 天天干天天操人体 | 久久视频99 | 国产一级特黄电影 | 国产精品女教师 | 国产视频精品在线 | 日韩免费成人 | 国产精品美女视频网站 | 久久99久久99精品免费看小说 | 亚洲性少妇性猛交wwww乱大交 | 亚洲免费观看视频 | 久久电影中文字幕视频 | 日本字幕网 | 美女福利视频一区二区 | 日韩专区在线 | 一区二区三区韩国免费中文网站 | 日韩一二区在线观看 | 亚洲综合爱| 成人av在线看 | 日韩av高清| 九九免费在线观看视频 | 最近能播放的中文字幕 | 成人动图 | 国内精品久久久 | 久久综合成人 | 黄色成人av | 国产成人一区二区三区 | 国产偷国产偷亚洲清高 | 亚洲综合小说 | 天堂入口网站 | 午夜视频免费在线观看 | 天天草天天干 | 亚洲精品白浆高清久久久久久 | 在线观看免费观看在线91 | 在线观看亚洲专区 | 国产精品精品久久久久久 | 国产精品欧美日韩 | 成人久久免费视频 | 国产亚洲精品久久19p | 亚洲精品在线看 | 日韩视频在线不卡 | 久久99精品波多结衣一区 | 91精品系列| 色在线高清 | 亚洲精品在线二区 | 国产在线无 | 最近高清中文字幕在线国语5 | 免费观看91视频大全 | 国产成人av在线影院 | www.com在线观看 | 日韩精品一区二区三区免费观看 | 亚洲国产日韩欧美 | 精品国产自在精品国产精野外直播 | 久久久精品一区二区三区 | 黄色在线免费观看网站 | 欧美国产亚洲精品久久久8v | 国产第页 | 久久视屏网 | 色姑娘综合天天 | 国产第一页在线播放 | 久久综合久久综合久久综合 | 99精品黄色片免费大全 | 久久免费视频网站 | 亚洲黄色免费在线看 | 亚洲综合五月 | 亚洲久草在线视频 | 欧美黑吊大战白妞欧美 | 久久这里只有精品久久 | av品善网| 99视频这里有精品 | 欧美 日韩 性 | 国产欧美日韩视频 | av 一区 二区 久久 | 黄色一级免费电影 | 免费在线观看成人 | 搡bbbb搡bbb视频 | 久久久久久麻豆 | 午夜久草| 天天干天天摸天天操 | 精品国产美女在线 | 国产午夜精品视频 | 久久精品国产第一区二区三区 | 天天色成人网 | 亚洲一级片av | 黄网站免费看 | 欧美综合久久 | 97视频播放 | 免费精品视频在线观看 | 91麻豆精品国产自产在线游戏 | 香蕉视频国产在线 | 中文字幕不卡在线88 | 日本久久中文字幕 | 久九视频 | 亚洲国产精品成人女人久久 | 久久99久久99精品免观看粉嫩 | 99精品偷拍视频一区二区三区 | 日韩在线资源 | 在线观看日本韩国电影 | 一本一本久久a久久精品牛牛影视 | 国产91探花| 婷婷av电影 | 国产视频欧美视频 | 伊人亚洲综合 | 综合色亚洲| 日韩三级免费观看 | 国产91学生粉嫩喷水 | 久久深夜 | 久久成人福利 | 中文字幕免费一区 | 另类五月激情 | 亚洲视频每日更新 | 人人插超碰 | 久在线观看| 521色香蕉网站在线观看 | 久久免费看片 | 免费在线成人 | 国产精品一区二区在线看 | 亚洲永久精品国产 | 成人小视频在线观看免费 | 黄色a在线观看 | 最新真实国产在线视频 | 久久国产热 | 精品久久久精品 | 手机av资源| 国产三级精品三级在线观看 | 精品久久久久久亚洲综合网站 | 激情六月婷婷久久 | 亚洲精品久久久久久久蜜桃 | 精品伦理一区二区三区 | 97电影院在线观看 | 久久色在线观看 | 久久久久久欧美二区电影网 | 亚洲天堂激情 | 一级α片免费看 | 国产二区精品 | 天天操天天是 | av电影不卡在线 | 婷婷成人在线 | 97超碰福利久久精品 | 在线看的av网站 | 国产亚洲综合在线 | 精品国产伦一区二区三区观看方式 | 97在线视频免费看 | 福利一区二区三区四区 | 日本精品一区二区在线观看 | 国内精品国产三级国产aⅴ久 | wwwwww色| 久久久国产精品人人片99精片欧美一 | 99久久久久国产精品免费 | 伊人狠狠色丁香婷婷综合 | 麻豆视频在线免费观看 | 国产精品福利在线观看 | 久久99久久99 | av大全在线免费观看 | 韩日精品中文字幕 | 黄色大片中国 | 国内精品久久久久影院优 | 国产亚洲成av人片在线观看桃 | 蜜臀aⅴ国产精品久久久国产 | 久久久精品 一区二区三区 国产99视频在线观看 | 日日草天天草 | 日操操| 国产精品欧美精品 | 欧美日韩一区三区 | 1区2区3区在线观看 三级动图 | 国产精品久久久av | 西西444www高清大胆 | av电影免费在线播放 | 天天射天天搞 | 五月开心综合 | 色综合欧洲 | 久久夜色电影 | 伊人资源站 | 成人久久 | 日韩欧美视频免费观看 | 国产美女在线精品免费观看 | 欧美日韩在线视频观看 | 欧美日韩一区二区三区不卡 | 国产免费又爽又刺激在线观看 | 中文字幕在线免费看线人 | av在线超碰 | 91精品资源 | 97超碰成人在线 | 久久精品麻豆 | 91看片淫黄大片91 | 色wwwww | 91成人免费 | 四虎国产视频 | 色婷婷视频在线 | 国产午夜av | 国产成人精品一区在线 | 日韩av成人免费看 | 欧美日本日韩aⅴ在线视频 插插插色综合 | 久久污视频 | 国产视频日韩视频欧美视频 | 国产色视频网站2 | 97成人在线观看视频 | 91视频麻豆视频 | 黄色免费网站下载 | 一区二区精品视频 | 涩涩网站在线观看 | 亚洲一级电影在线观看 | 精品久久久国产 | 在线免费av播放 | 91九色国产 | 一级大片在线观看 | 日本久久久亚洲精品 | 在线 国产 亚洲 欧美 | 欧美成人精品欧美一级乱 | 国产欧美精品一区aⅴ影院 99视频国产精品免费观看 | 玖玖爱免费视频 | 韩国三级av在线 | 99亚洲精品视频 | 国产一级黄色av | 久久国产精品99久久久久久老狼 | 五月婷婷在线综合 | 欧美日韩成人一区 | 在线激情av电影 | 国产在线看一区 | 国产福利精品在线观看 | 99电影| 香蕉视频在线免费 | 日韩欧美电影在线观看 | 久久久久精 | 日本老少交 | 亚洲国产合集 | 国产精品久久久久久a | 在线观看日本高清mv视频 | 在线天堂v | 91综合久久一区二区 | 天天干天天操人体 | 日本精品在线看 | 久久精品视频免费观看 | 中文字幕在线观看一区二区 | 色资源中文字幕 | 精品国产1区 | 天天色影院 | 亚洲 欧美 变态 国产 另类 | 一区三区视频在线观看 | 亚洲天堂网站视频 | 五月婷久久 | 久久99国产精品二区护士 | 亚洲精品国偷拍自产在线观看蜜桃 | 在线观看免费中文字幕 | 天天艹| 五月激情六月丁香 | 91九色蝌蚪在线 | 色永久免费视频 | 亚洲国产精品视频在线观看 | 九九热在线精品视频 | 亚洲精品国久久99热 | 夜夜夜夜爽 | 五月激情姐姐 | 国产aa精品 | 精品伦理一区二区三区 | 日日夜夜综合 | 日韩精品不卡在线 | 精品国产乱码久久久久久1区二区 | 在线观看91精品国产网站 | 国产在线一区观看 | 色一级片 | 国产美女视频网站 | www夜夜 | 国产香蕉久久精品综合网 | 亚洲精品视频中文字幕 | 激情综合五月 | 狠狠操操| 久久99久久99精品免观看软件 | 亚洲精品乱码久久久久久蜜桃欧美 | 国产免费一区二区三区最新 | 欧美在线视频二区 | 国产精品99久久久精品 | 国产综合精品一区二区三区 | 亚洲综合网站在线观看 | 综合网色 | 日日夜夜精品网站 | 国产一二三四在线观看视频 | 欧美日韩免费网站 | 麻豆91网站| 狠狠狠狠狠狠狠干 | 日本黄色一级电影 | 高清不卡毛片 | 免费观看一级成人毛片 | 91九色在线视频 | 在线免费观看黄色 | 最新的av网站 | 午夜影视一区 | 日韩av影视在线 | 在线国产精品一区 | 久久久麻豆| 日本黄色免费大片 | 天天操天天爽天天干 | 日本性xxx| 亚洲 中文 在线 精品 | 国产一二三四在线观看视频 | 亚洲小视频在线观看 | 一区二区三区在线不卡 | 草久久精品 | 伊人永久在线 | av片中文| 一级黄色大片在线观看 | 又色又爽又黄 | 新版资源中文在线观看 | 偷拍精偷拍精品欧洲亚洲网站 | 波多野结衣在线视频一区 | www黄免费| 国产性天天综合网 | 区一区二在线 | 超碰在97 | 欧美黑人猛交 | 在线观看完整版 | 免费看久久 | 视频二区在线 | 国产精品一区二区av日韩在线 | 在线三级播放 | 久久精品99久久久久久2456 | 中文字幕在线观看视频一区 | 亚洲第一香蕉视频 | 国产一区高清在线 | 激情视频在线高清看 | 成人免费一级 | av色一区 | 色国产视频 | 在线视频 国产 日韩 | 亚洲精品五月 | 国产色中涩 | 久久天天躁夜夜躁狠狠躁2022 | 色99之美女主播在线视频 | 91av在线播放视频 | 国产精品欧美日韩在线观看 | 亚洲三级毛片 | 国产第一福利网 | 在线观看久草 | 黄色av电影在线 | 西西444www大胆高清视频 | 五月婷婷丁香 | 国内精品福利视频 | 欧美aa一级片 | 精品国产一区二区三区日日嗨 | 在线视频欧美精品 | 国产中文欧美日韩在线 | 亚洲国产片色 | 精品国产123 | 国产免费久久久久 | 久久综合狠狠综合久久狠狠色综合 | 久草综合在线观看 | 又黄又爽免费视频 | 久久久成人精品 | 国产成人在线综合 | 日韩精品一区二区三区中文字幕 | 欧美激情在线看 | 丁香在线视频 | 欧美另类成人 | 国产亚洲精品久久久久动 | 久久综合久色欧美综合狠狠 | 国产午夜精品一区二区三区在线观看 | 欧美日韩免费在线观看视频 | 亚洲综合色丁香婷婷六月图片 | www.看片网站 | 久久精品亚洲一区二区三区观看模式 | 成人在线视频论坛 | 国产精品综合在线观看 | 天天操天天操一操 | 四川bbb搡bbb爽爽视频 | 热久在线 | 国产成人精品久久久 | 国产成人亚洲在线观看 | 日韩av手机在线看 | 久久视频国产精品免费视频在线 | 免费色视频网站 | 婷婷久草 | 国产精品激情在线观看 | 精品国产色 | 欧美性黑人 | 亚洲综合视频在线 | 国产日本在线观看 | www,黄视频| 97在线看| 在线免费日韩 | 久久精品中文字幕少妇 | 91手机电影 | 国产精品一区二区三区在线免费观看 | 日韩免费三级 | 欧美激情综合五月色丁香 | 国产成人精品国内自产拍免费看 | 天天色视频 | 中文字幕在线免费看 | 最近av在线 | adn—256中文在线观看 | 视频三区| 国产精品久久久电影 | 国产精品一区二区久久精品 | 18女毛片 | 国产黄色成人av | 婷婷久久丁香 | 久久久久久久av麻豆果冻 | 国产视频一区二区在线观看 | 五月天开心| 99精品小视频 | 久久99这里只有精品 | 视频在线观看入口黄最新永久免费国产 | 999久久久久久久久6666 | 国产999视频在线观看 | 国产亚洲免费观看 | 久久视频在线观看中文字幕 | 特黄特黄的视频 | 狠狠色丁香婷综合久久 | 天天操天天爽天天干 | 免费亚洲精品视频 | 中文字幕在线观看第一区 | av免费片 | 久久人人97超碰精品888 | 久久久99精品免费观看 | 国产精品久久久久av福利动漫 | 国产不卡在线观看 | 91视频中文字幕 | 97在线免费观看视频 | 超碰97人人干 | 亚洲91视频 | 亚州激情视频 | 人人澡人人添人人爽一区二区 | 精品亚洲一区二区 | 黄网站色欧美视频 | 9999精品视频 | 久久久精品久久日韩一区综合 | 久久超级碰| 国产一级视频在线 | 免费久草视频 | 久香蕉 | 天堂网一区二区三区 | 久久久久免费精品视频 | 日韩一区二区三区在线观看 | 国产黄色片一级 | 久久精品国产亚洲aⅴ | 精品人人人 | avwww在线观看 | 91麻豆传媒| 日韩欧美在线观看一区二区三区 | 国产一区二区不卡在线 | 国产裸体视频网站 | 夜夜躁天天躁很躁波 | 久久精品视频日本 | 亚洲区另类春色综合小说 | 成人黄色电影在线播放 | 日韩videos高潮hd| 日韩va在线观看 | 欧美日韩视频在线观看一区二区 | 日本最大色倩网站www | 久久爱资源网 | 国产精品高 | a v在线观看 | 久久久黄视频 | 91精品久久久久久综合五月天 | 欧美日韩国产一区二区三区在线观看 | 国产美腿白丝袜足在线av | 99久久久国产精品免费观看 | 国产亚洲精品久久久久久 | 国产高清在线观看 | 亚洲网站在线 | 日日操天天操狠狠操 | 四虎国产 | 国产无区一区二区三麻豆 | 91在线免费观看国产 | 99色精品视频 | 欧美孕交vivoestv另类 | 欧美日韩在线播放 | 久久久久久高潮国产精品视 | 91视频xxxx| 夜夜澡人模人人添人人看 | 国产精品原创在线 | 中文字幕国产精品一区二区 | 97视频在线观看网址 | 婷婷色综 | 91九色老| 五月天久久久久久 | 国产欧美精品一区二区三区 | 国产午夜精品免费一区二区三区视频 | 亚洲一区av | 久草在线官网 | 99久久激情视频 | 麻豆一区在线观看 | 中字幕视频在线永久在线观看免费 | 在线观看日韩中文字幕 | 精品日韩中文字幕 | 国产主播大尺度精品福利免费 | 97福利在线观看 | 一区二区三区在线观看 | 日韩免费区| 97国产情侣爱久久免费观看 | 午夜av色 | 日韩一二三| 国内精品久久久久久久影视麻豆 | 精品久久电影 | 美女黄视频免费 | 免费一级日韩欧美性大片 | 欧美最爽乱淫视频播放 | 久久久久国产一区二区三区 | 国产剧情在线一区 | 天天亚洲综合 | 中文字字幕在线 | 中文字幕国语官网在线视频 | 91麻豆操| 亚洲精品欧美专区 | 日韩av一区二区三区在线观看 | 日韩一区二区免费在线观看 | 91爱爱免费观看 | 中文字幕在线播放日韩 | 日韩影视精品 | 国产精品美女久久久久久网站 | 国产精品网红福利 | 日本在线精品视频 | 久久大片 | 中文字幕字幕中文 | 天天操天天操天天操天天操天天操天天操 | 成人黄色电影视频 | 最新av电影网站 | 色91在线视频 | 91在线蜜桃臀 | 日韩伦理片hd | 国产精品色 | 精品在线看 | 亚洲成av人片在线观看www | 国产中文字幕免费 | 中文字幕 国产精品 | 国内久久久久 | 免费观看国产成人 | 亚洲激情电影在线 | 久久久免费精品 | 日韩极品在线 | 狠狠色丁香 | 美女在线观看av | 免费福利在线 | 91成人免费在线视频 | www.黄色| 久久久久久久久电影 | 欧美亚洲成人免费 | 国产精品美女久久久久久久 | 一区二区三区精品久久久 | 精品视频网站 | 国产精彩在线视频 | 天天插视频 | 亚洲一级免费电影 | 999在线视频 | 国产精品中文字幕在线 | 在线看片成人 | 色91在线视频 | 综合在线观看 | 香蕉在线视频观看 | 日韩高清不卡在线 | 日韩久久久| 热久久免费视频精品 | 免费在线观看日韩视频 | 青青久视频 | 黄色资源在线 | 麻豆观看| av直接看| 国产精品久久久免费 | 成人av网址大全 | 国产中文字幕视频 | 丰满少妇麻豆av | 国内成人精品2018免费看 | 欧美最猛性xxxxx亚洲精品 | 精品国产综合区久久久久久 | 99久久精品国产亚洲 | 久久黄色美女 | 国产一区二区不卡视频 | 久久黄视频 | 97在线公开视频 | 中文字幕国产在线 | 中文字幕在线播放一区二区 | 9久久精品| 综合久久久久 | 日韩三区在线 | 亚洲撸撸 | 亚洲黄色小说网 | 韩日精品在线 | 精品亚洲免a | 国产成人av一区二区三区在线观看 | 黄色在线观看污 | 91看片一区二区三区 | 最近更新的中文字幕 | 91亚色在线观看 | 欧美国产一区在线 | 国产一级久久 | 成人午夜精品福利免费 | 国产在线一区二区 | 亚洲精品 在线视频 | 国产美女精彩久久 | 久久久毛片 | 丁香在线| 免费看av片网站 | 91系列在线 | 成年人在线免费看视频 | 超黄视频网站 | 四虎天堂 | av线上免费观看 | 成+人+色综合 | 日韩av高清在线观看 | 欧美日韩视频一区二区 | 欧美91精品国产自产 | 天天干天天拍天天操 | 久久精品黄 | 麻豆视频网址 | 97精品国产91久久久久久久 | 久久久视屏 | 色射爱| 韩日av一区二区 | 天天射天天干天天爽 | 成人网在线免费视频 | 色狠狠操 | 激情五月五月婷婷 | 肉色欧美久久久久久久免费看 | 亚洲精品永久免费视频 | 亚洲午夜不卡 | 麻豆观看 | 久草在线视频免费资源观看 | 欧美一级片在线观看视频 | 欧美婷婷色 | 精品国产乱码久久久久久三级人 | 国产偷v国产偷∨精品视频 在线草 | 日韩精品一区二区在线 | 成年人免费电影在线观看 | www五月天婷婷 | 99看视频在线观看 | 色小说av | 国产v在线 | 综合天堂av久久久久久久 | 日日操操 | www.黄色网.com | 91自拍成人 | 在线91色| 国产不卡在线播放 | 日韩在线播放视频 | 午夜影院日本 | 免费国产在线精品 | 美女视频黄是免费的 | 欧美日韩视频 | 国产亚洲一区二区三区 | 亚洲国产成人精品久久 | 天天拍天天色 | 在线观看日韩精品视频 | 亚洲区二区 | 欧美色噜噜 | 激情欧美在线观看 | 狠狠色香婷婷久久亚洲精品 | 国产精品少妇 | 黄色av一区二区三区 | 亚洲精品大全 | 波多野结衣久久精品 | 午夜神马福利 | 在线视频欧美亚洲 | 久久久久精 | 国产精品99久久久久 | 国产91精品一区二区麻豆网站 | 国产小视频在线观看 | 亚洲欧美视频一区二区三区 | 91亚洲国产成人 | 免费精品国产va自在自线 | 深夜福利视频在线观看 | 日日婷婷夜日日天干 | 日本在线视频网址 | 久久深爱网 | 国产日产亚洲精华av | 成人日批视频 | 精品二区久久 | 午夜视频免费在线观看 | 一本一本久久a久久精品综合妖精 | 国产精品美女久久久久久久网站 | 精品久久一区二区三区 | 日韩欧美专区 | 色狠狠狠 | 国产一级二级三级在线观看 | 成人黄色片免费 | 99精品国产99久久久久久97 | 国产高清免费视频 | 在线免费色 | 色a资源在线 | 久久综合狠狠综合久久狠狠色综合 | 日韩高清在线一区 | 91av免费看 | 亚洲午夜激情网 | 国产一区二区手机在线观看 | av在线电影播放 | 久久久精品 一区二区三区 国产99视频在线观看 | 黄色免费av | 日韩r级电影在线观看 | 欧美三级高清 | 又长又大又黑又粗欧美 | www免费看片com | 国产精品白丝av | av在线播放一区二区三区 | 亚洲精品456在线播放 | 狠狠躁日日躁 | 国产精品二区三区 | 婷婷六月丁 | 热九九精品| 在线va网站 | 国产精品久久久久久久久软件 | 久久69精品| av免费在线观 | 国产精品视频专区 | 久久夜av | 国产99视频在线观看 | 在线v| 亚洲国产成人久久综合 | 久草在线视频在线 | 91免费高清 | 国产精品免费大片视频 | 成人观看 | 97精品国产91久久久久久久 | 在线91精品 | 91mv.cool在线观看 | 国内精品国产三级国产aⅴ久 | 黄色片亚洲 | 亚洲激情 | 97超在线| 蜜臀av性久久久久蜜臀aⅴ四虎 | 欧美日韩国内在线 | 久久99国产精品免费网站 | 日本黄色免费播放 | 国产美腿白丝袜足在线av | 国产成人一区二区三区影院在线 | 亚洲一级黄色大片 | 97精品在线观看 | 欧美亚洲久久 | 色婷在线 | 亚洲日韩精品欧美一区二区 | 婷婷久久婷婷 | 四川bbb搡bbb爽爽视频 | 在线亚洲观看 | 欧美大片www | 亚洲精品美女久久久 | 国产精品久久电影网 | av高清免费在线 | 精品国产乱码久久久久久三级人 | 2018亚洲男人天堂 | 免费观看xxxx9999片 | 成人av电影免费 | 最新婷婷色| 久久好看免费视频 | 在线a人片免费观看视频 | 国产男女免费完整视频 | 免费在线观看成人 | 97看片| 亚洲资源片| 国产五月婷婷 | 国产一区二区三区在线 | 久久综合九色综合久久久精品综合 | 天天色天天射综合网 | 色成人亚洲 | 五月天亚洲激情 | 亚洲精品一区二区三区高潮 | 在线观看中文字幕2021 | 成人av在线直播 | 国产精品国产三级国产不产一地 | 久草网在线观看 | 九九在线国产视频 | www.亚洲在线| 亚洲日韩中文字幕在线播放 | 制服丝袜一区二区 | 国产高清在线一区 | 狠狠干综合 | 四虎伊人| 精品国产91亚洲一区二区三区www | 国内精品久久久久久久久久久 | 激情五月婷婷网 | 欧美 日韩 视频 | 中文字幕色婷婷在线视频 | 99久久综合国产精品二区 | 亚洲精品乱码久久久久久按摩 | 人人射人人爽 | 亚洲伦理一区 | 免费a v观看| 午夜电影中文字幕 | 日本在线中文在线 | 成年人黄色免费视频 | 中文乱码视频在线观看 | 五月婷婷深开心 | 日本韩国精品在线 |