《電子技術應用》
您所在的位置:首頁 > 通信與網絡 > 設計應用 > Linux網絡驅動程序功能分析
Linux網絡驅動程序功能分析
中電網
摘要: 如果網絡設備(包括wireless)是PCI規范的,則先是向內核注冊該PCI設備(pci_register_driver),然后由pci_driver數據結構中的probe函數指針所指向的偵測函數來初始化該PCI設備,并且同時注冊和初始化該網絡設備。
關鍵詞: Linux 網絡驅動 PCI
Abstract:
Key words :

1.驅動模塊的加載和卸載

如果網絡設備(包括wireless)是PCI規范的,則先是向內核注冊該PCI設備(pci_register_driver),然后由pci_driver數據結構中的probe函數指針所指向的偵測函數來初始化該PCI設備,并且同時注冊和初始化該網絡設備。

如果網絡設備(包括wireless)是PCMCIA規范的,則先是向內核注冊該PCMCIA設備(register_pccard_driver),然后driver_info_t數據結構中的attach函數指針所指向的偵測函數來初始化該PCMCIA設備,并且同時注冊和初始化該網絡設備。

static int __init tg3_init(void)

{

//先注冊成PCI設備,并初始化,如果是其他的ESIA,PCMCIA,用其他函數

return pci_module_init(&tg3_driver);

}

static void __exit tg3_cleanup(void)

{

pci_unregister_driver(&tg3_driver);//注銷PCI設備

}

module_init(tg3_init); //驅動模塊的加載

module_exit(tg3_cleanup); //驅動模塊的卸載

申明為PCI設備:

static struct pci_driver tg3_driver = {

.name = DRV_MODULE_NAME,

.id_table = tg3_pci_tbl, //此驅動所支持的網卡系列,vendor_id, device_id

.probe = tg3_init_one, //初始化網絡設備的回調函數

.remove = __devexit_p(tg3_remove_one), //注銷網絡設備的回調函數

.suspend = tg3_suspend, //設備掛起函數

.resume = tg3_resume //設備恢復函數

};

2.PCI設備探測函數probe,初始化網絡設備

static int __devinit tg3_init_one(struct pci_dev *pdev, const struct pci_device_id *ent)

{

//初始化設備,使I/O,memory可用,喚醒設備

pci_enable_device(pdev);

//申請內存空間,配置網卡的I/O,memory資源

pci_request_regions(pdev, DRV_MODULE_NAME);

pci_set_master(pdev);

//設置DMA屬性

pci_set_dma_mask(pdev, (u64) 0xffffffffffffffff);

//網卡 I/O,memory資源的啟始地址

tg3reg_base = pci_resource_start(pdev, 0);

//網卡I/O,memory資源的大小

tg3reg_len = pci_resource_len(pdev, 0);

//分配并設置網絡設備

dev = alloc_etherdev(sizeof(*tp));

//申明為內核設備模塊

SET_MODULE_OWNER(dev);

//初始化私有結構中的各成員值

tp = dev->priv;

tp->pdev = pdev;

tp->dev = dev;

……

//鎖的初始化

spin_lock_init(&tp->lock);

//映射I/O,memory地址到私有域中的寄存器結構

tp->regs = (unsigned long) ioremap(tg3reg_base, tg3reg_len);

dev->irq = pdev->irq;

//網絡設備回調函數賦值

dev->open = tg3_open;

dev->stop = tg3_close;

dev->get_stats = tg3_get_stats;

dev->set_multicast_list = tg3_set_rx_mode;

dev->set_mac_aDDRess = tg3_set_mac_addr;

dev->do_ioctl = tg3_ioctl;

dev->tx_timeout = tg3_tx_timeout;

dev->hard_start_xmit= tg3_start_xmit;

//網卡的MAC地址賦值dev->addr

tg3_get_device_address(tp);

//注冊網絡設備

register_netdev(dev);

//把網絡設備指針地址放入PCI設備中的設備指針中

pci_set_drvdata(pdev, dev);

}

3.注銷網絡設備

static void __devexit tg3_remove_one(struct pci_dev *pdev)

{

struct net_device *dev = pci_get_drvdata(pdev);

//注銷網絡設備

unregister_netdev(dev);

//取消地址映射

iounmap((void *) ((struct tg3 *)(dev->priv))->regs);

//釋放網絡設備

kfree(dev);

//釋放PCI資源

pci_release_regions(pdev);

//停用PCI設備

pci_disable_device(pdev);

//PCI設備中的設備指針賦空

pci_set_drvdata(pdev, NULL);

}

4.打開網絡設備

static int tg3_open(struct net_device *dev)

{

//分配一個中斷

request_irq(dev->irq, tg3_interrupt, SA_SHIRQ, dev->name, dev);

/* int request_irq(unsigned int irq,

void (*handler)(int irq, void *dev_id, struct pt_regs *regs),

unsigned long irqflags,

const char * devname,

void *dev_id);

irq是要申請的硬件中斷號。在Intel平臺,范圍0--15。handler是向系統登記的中斷處理函數。這是一個回調函數,中斷發生時,系統調用這個函數,傳入的參數包括硬件中斷號,device id,寄存器值。dev_id就是下面的request_irq時傳遞給系統的參數dev_id。irqflags是中斷處理的一些屬性。比較重要的有SA_INTERRUPT,標明中斷處理程序是快速處理程序(設置SA_INTERRUPT)還是慢速處理程序(不設置SA_INTERRUPT)。快速處理程序被調用時屏蔽所有中斷。慢速處理程序不屏蔽。還有一個SA_SHIRQ屬性,設置了以后運行多個設備共享中斷。dev_id在中斷共享時會用到。一般設置為這個設備的device結構本身或者NULL。中斷處理程序可以用dev_id找到相應的控制這個中斷的設備,或者用rq2dev_map找到中斷對應的設備。*/

//初始化硬件

tg3_init_hw(tp);

//初始化收包和發包的緩沖區

tg3_init_rings(tp);

//初始化定時器

init_timer(&tp->timer);

tp->timer.expires = jiffies + tp->timer_offset;

tp->timer.data = (unsigned long) tp;

tp->timer.function = tg3_timer; //超時回調函數

add_timer(&tp->timer);

//允許網卡開始傳輸包

netif_start_queue(dev);

}

5.關閉網絡設備

static int tg3_close(struct net_device *dev)

{

//停止網卡傳輸包

netif_stop_queue(dev);

netif_carrier_off(tp->dev);

//去除定時器

del_timer_sync(&tp->timer);

//釋放收包和發包的緩沖區

tg3_free_rings(tp);

//釋放中斷

free_irq(dev->irq, dev);

}

[NextPage]

6.硬件處理數據包發送

static int tg3_start_xmit(struct sk_buff *skb, struct net_device *dev)

{

len = (skb->len - skb->data_len);

//以DMA方式向網卡物理設備傳輸包。如果是wireless的話,需要根據802.11協議及硬件的規范從新填充

//硬件幀頭,然后提交給硬件發送。

mapping = pci_map_single(tp->pdev, skb->data, len, PCI_DMA_TODEVICE);

tp->tx_buffers[entry].skb = skb;

pci_unmap_addr_set(&tp->tx_buffers[entry], mapping, mapping);

//硬件發送

tg3_set_txd(tp, entry, mapping, len, base_flags, mss_and_is_end);

//記錄發包開始時間

dev->trans_start = jiffies;

}

7.中斷處理收包,發包

static void tg3_interrupt(int irq, void *dev_id, struct pt_regs *regs)

{

//如果要收包

tg3_rx(tp);

//如果要發包

tg3_tx(tp);

}

8.發包

static void tg3_tx(struct tg3 *tp)

{

struct tx_ring_info *ri = &tp->tx_buffers[sw_idx];

struct sk_buff *skb = ri->skb;

//以DMA方式向網卡傳輸包完畢

pci_unmap_single(tp->pdev, pci_unmap_addr(ri, mapping),

(skb->len - skb->data_len), PCI_DMA_TODEVICE);

ri->skb = NULL;

dev_kfree_skb_irq(skb);

}

9.收包

static int tg3_rx(struct tg3 *tp, int budget)

{

struct sk_buff *copy_skb;

//分配一個包

copy_skb = dev_alloc_skb(len + 2);

copy_skb->dev = tp->dev;

//修改包頭空間

skb_reserve(copy_skb, 2);

//加入數據到包中

skb_put(copy_skb, len);

//以DMA方式從網卡傳輸回數據

pci_dma_sync_single(tp->pdev, dma_addr, len, PCI_DMA_FROMDEVICE);

memcpy(copy_skb->data, skb->data, len);

skb = copy_skb;

 //解析包的協議

skb->protocol = eth_type_trans(skb, tp->dev);

//把包送到協議層

netif_rx(skb);

//記錄收包時間

tp->dev->last_rx = jiffies;

}

10.讀取包的網卡收發包的狀態,統計數據

static struct net_device_stats *tg3_get_stats(struct net_device *dev)

{

//從硬件相關的寄存器讀取數據,累加

//stats->rx_packets, stats->tx_packets, stats->rx_bytes, stats->tx_bytes等

}

11.用戶的ioctl命令系統調用

static int tg3_ioctl(struct net_device *dev, struct ifreq *ifr, int cmd)

{

struct mii_ioctl_data *data = (struct mii_ioctl_data *)&ifr->ifr_data;

switch(cmd) {

//ethtool程序命令的調用

case SIO*HTOOL:

return tg3_ethtool_ioctl(dev, (void *) ifr->ifr_data);

//mii程序命令的調用

case SIOCGMIIREG: {

err = tg3_readphy(tp, data->reg_num & 0x1f, &mii_regval)

data->val_out = mii_regval;

return err;

}

……

}

}

12.PCI設備的掛起和恢復函數

static int tg3_suspend(struct pci_dev *pdev, u32 state)

{

//停用網卡的中斷寄存器

tg3_disable_ints(tp);

//停止網卡收發包

netif_device_detach(dev);

//停止網卡某些硬件,fireware的一些功能

tg3_halt(tp);

//設置網卡的電源狀態

tg3_set_power_state(tp, state);

}

static int tg3_resume(struct pci_dev *pdev)

{

//恢復網卡電源

tg3_set_power_state(tp, 0);

//允許網卡收發包

netif_device_attach(dev);

//初始化收發包的緩沖區

tg3_init_rings(tp);

//初始化網卡硬件

tg3_init_hw(tp);

//打開網卡中斷寄存器

tg3_enable_ints(tp);

}

13.參數設置

在驅動程序里還提供一些方法供系統對設備的參數進行設置和讀取信息。一般只有超級用戶(root)權限才能對設備參數進行設置。設置方法有:

tg3_set_mac_addr (dev->set_mac_address)

當用戶調用ioctl類型為SIOCSIFHWADDR時是要設置這個設備的mac地址。一般對mac地址的設置沒有太大意義的。

dev->set_config()

當用戶調用ioctl時類型為SIOCSIFMAP時,系統會調用驅動程序的set_config方法

用戶會傳遞一個ifmap結構包含需要的I/O、中斷等參數。

總結:

所有的Linux網絡驅動程序遵循通用的接口。設計時采用的是面向對象的方法。一個設備就是一個對象(net_device 結構),它內部有自己的數據和方法。一個網絡設備最基本的方法有初始化,發送和接收。

Linux網絡驅動程序的體系結構可以劃分為四層:

網絡協議接口,網絡設備接口,設備驅動功能,網絡設備和網絡媒介層

網絡驅動程序,最主要的工作就是完成設備驅動功能層。在Linux中所有網絡設備都抽象為一個接口,這個接口提供了對所有網絡設備的操作集合。由數據結構struct net_device來表示網絡設備在內核中的運行情況,即網絡設備接口。它既包括純軟件網絡設備接口,如環路(Loopback),也包括硬件網絡設備接口,如以太網卡。而由以dev_base為頭指針的設備鏈表來集體管理所有網絡設備,該設備鏈表中的每個元素代表一個網絡設備接口。數據結構net_device中有很多供系統訪問和協議層調用的設備方法,包括初始化,打開和關閉網絡設備的open和stop函數,處理數據包發送的hard_start_xmit函數,以及中斷處理函數等。

網絡設備在Linux里做專門的處理。Linux的網絡系統主要是基于BSD unix的socket機制。在系統和驅動程序之間定義有專門的數據結構(sk_buff)進行數據的傳遞。系統里支持對發送數據和接收數據的緩存,提供流量控制機制,提供對多協議的支持。

此內容為AET網站原創,未經授權禁止轉載。
亚洲一区二区欧美_亚洲丝袜一区_99re亚洲国产精品_日韩亚洲一区二区
在线一区二区三区四区五区| 午夜精品999| 国产精品一区在线观看你懂的| 欧美福利小视频| 免费成人高清在线视频| 久久免费视频网| 久久精品国产一区二区三| 亚洲欧美一区二区三区久久| 亚洲一区二区三区四区五区午夜| av不卡在线看| 99精品欧美一区二区蜜桃免费| 亚洲国产精品va在看黑人| 久久高清福利视频| 久久国产精品第一页| 欧美有码在线观看视频| 亚洲欧美在线免费观看| 亚洲欧美日韩一区在线| 欧美一级大片在线免费观看| 欧美一区永久视频免费观看| 欧美有码在线视频| 亚洲国产精品999| 最新国产成人av网站网址麻豆 | 亚洲欧美在线磁力| 香蕉成人伊视频在线观看| 亚欧美中日韩视频| 久久成人一区| 亚洲美女色禁图| 一区二区三区蜜桃网| 亚洲综合色婷婷| 久久成人18免费观看| 久久精品日产第一区二区三区| 久久免费视频在线观看| 欧美高清一区二区| 欧美日韩一区二区在线观看视频| 欧美午夜剧场| 国产亚洲精品v| 亚洲第一黄色| 亚洲最新中文字幕| 亚洲午夜高清视频| 欧美一级免费视频| 亚洲精品美女久久久久| 亚洲一区二区综合| 欧美一区2区三区4区公司二百 | 国产最新精品精品你懂的| 亚洲大胆女人| 妖精视频成人观看www| 欧美亚洲系列| 91久久综合亚洲鲁鲁五月天| 一区二区三区四区精品| 午夜精品在线| 另类专区欧美制服同性| 欧美日韩成人免费| 国产精品一区久久久久| 激情综合自拍| 99re8这里有精品热视频免费| 亚洲自拍偷拍麻豆| 亚洲精品久久视频| 欧美一级一区| 欧美激情一区二区久久久| 国产精品麻豆va在线播放| 激情综合自拍| 亚洲一级一区| 亚洲乱码国产乱码精品精可以看| 午夜国产精品影院在线观看| 卡通动漫国产精品| 国产精品久久久亚洲一区| 亚洲成人资源| 欧美亚洲一区二区在线| 一区二区福利| 毛片基地黄久久久久久天堂| 国产精品v欧美精品∨日韩| 极品裸体白嫩激情啪啪国产精品| 一区二区三区精品视频在线观看| 亚洲大片av| 午夜在线一区二区| 欧美欧美全黄| 精品91久久久久| 亚洲在线视频一区| 一本色道精品久久一区二区三区| 久久riav二区三区| 欧美三级特黄| 亚洲国产毛片完整版| 香蕉久久夜色精品国产使用方法 | 久久久91精品国产一区二区精品| 欧美日韩国产精品成人| 狠狠爱综合网| 亚洲欧美日韩系列| 亚洲午夜91| 欧美精品日韩综合在线| 黄色日韩精品| 欧美一级视频精品观看| 亚洲欧美日韩一区| 欧美日韩在线三区| 亚洲国产精品一区制服丝袜 | 一区二区三区高清在线| 亚洲精品一二区| 久久综合狠狠综合久久综青草| 国产精品亚洲网站| 宅男精品视频| 中国成人在线视频| 欧美大片网址| 玉米视频成人免费看| 欧美一区二区三区在线免费观看| 亚洲欧美成人在线| 欧美日韩综合一区| 日韩一级免费| 一本综合久久| 欧美日韩国产a| 亚洲美女精品久久| 99精品福利视频| 欧美国产一区二区| 亚洲狠狠婷婷| 亚洲免费观看高清完整版在线观看| 久热精品视频| 极品少妇一区二区| 亚洲高清自拍| 老司机成人网| 尤物视频一区二区| 亚洲激情在线| 欧美成人一区二区| 亚洲激情在线激情| 亚洲理论在线观看| 欧美精选午夜久久久乱码6080| 精品白丝av| 亚洲激情成人在线| 欧美.com| 亚洲人成网站在线播| 亚洲精品女av网站| 欧美日韩大片| 一本到高清视频免费精品| 亚洲午夜激情网站| 国产精品久久久久久久久久ktv| 在线综合+亚洲+欧美中文字幕| 亚洲在线中文字幕| 国产精品嫩草久久久久| 亚洲欧美另类在线| 久久精品中文字幕免费mv| 红桃av永久久久| 亚洲精品乱码久久久久久按摩观| 欧美精品久久一区二区| 日韩一二三在线视频播| 午夜精品美女久久久久av福利| 国产精品三上| 欧美亚洲色图校园春色| 久久嫩草精品久久久精品一| 在线成人小视频| 一区二区三区蜜桃网| 国产精品视频| 久久狠狠婷婷| 欧美美女操人视频| 亚洲一区二区三区777| 欧美在线观看网站| 狠狠色狠狠色综合系列| 亚洲精品免费一区二区三区| 欧美三级电影精品| 午夜亚洲激情| 蜜桃av一区二区三区| 夜夜嗨av一区二区三区免费区| 亚洲欧美文学| 精品不卡在线| 一区二区三区日韩| 国产精品尤物| 亚洲日本在线视频观看| 欧美体内she精视频| 小黄鸭精品密入口导航| 欧美福利网址| 亚洲欧美日韩国产| 免费久久久一本精品久久区| 亚洲乱码久久| 久久精品国产免费| 亚洲欧洲精品一区二区三区 | 久久久一二三| 亚洲精品乱码久久久久久黑人| 亚洲欧美区自拍先锋| 狠狠色丁香久久婷婷综合丁香| 夜夜精品视频| 国产亚洲成精品久久| 亚洲精品久久久久久下一站| 国产精品视频xxx| 亚洲茄子视频| 国产精品羞羞答答| 亚洲精品字幕| 国产欧美日本| aa亚洲婷婷| 一区二区亚洲精品| 午夜亚洲福利在线老司机| 亚洲二区在线视频| 欧美亚洲尤物久久| 亚洲九九爱视频| 久久综合一区二区三区| 亚洲视频一区在线观看| 美女视频黄a大片欧美| 中文精品99久久国产香蕉| 免费成人黄色av| 午夜一区不卡| 欧美午夜在线| 亚洲麻豆av| 激情欧美一区二区三区| 亚洲欧美日韩国产| 亚洲精品影院在线观看|