<meter id="pryje"><nav id="pryje"><delect id="pryje"></delect></nav></meter>
          <label id="pryje"></label>

          新聞中心

          EEPW首頁 > 嵌入式系統(tǒng) > 設計應用 > Linux網(wǎng)絡驅(qū)動程序

          Linux網(wǎng)絡驅(qū)動程序

          作者: 時間:2010-08-25 來源:網(wǎng)絡 收藏

            1.驅(qū)動模塊的加載和卸載

          本文引用地址:http://www.ex-cimer.com/article/151624.htm

            如果設備(包括wireless)是PCI規(guī)范的,則先是向內(nèi)核注冊該PCI設備(pci_register_driver),然后由pci_driver數(shù)據(jù)結(jié)構(gòu)中的probe函數(shù)指針所指向的偵測函數(shù)來初始化該PCI設備,并且同時注冊和初始化該設備。

            如果設備(包括wireless)是PCMCIA規(guī)范的,則先是向內(nèi)核注冊該PCMCIA設備(register_pccard_driver),然后driver_info_t數(shù)據(jù)結(jié)構(gòu)中的attach函數(shù)指針所指向的偵測函數(shù)來初始化該PCMCIA設備,并且同時注冊和初始化該網(wǎng)絡設備。

            static int __init tg3_init(void)

            {

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

            return pci_module_init(tg3_driver);

            }

            static void __exit tg3_cleanup(void)

            {

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

            }

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

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

            申明為PCI設備:

            static struct pci_driver tg3_driver = {

            .name = DRV_MODULE_NAME,

            .id_table = tg3_pci_tbl, //此驅(qū)動所支持的網(wǎng)卡系列,vendor_id, device_id

            .probe = tg3_init_one, //初始化網(wǎng)絡設備的回調(diào)函數(shù)

            .remove = __devexit_p(tg3_remove_one), //注銷網(wǎng)絡設備的回調(diào)函數(shù)

            .suspend = tg3_suspend, //設備掛起函數(shù)

            .resume = tg3_resume //設備恢復函數(shù)

            };

            2.PCI設備探測函數(shù)probe,初始化網(wǎng)絡設備

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

            {

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

            pci_enable_device(pdev);

            //申請內(nèi)存空間,配置網(wǎng)卡的I/O,memory資源

            pci_request_regions(pdev, DRV_MODULE_NAME);

            pci_set_master(pdev);

            //設置DMA屬性

            pci_set_dma_mask(pdev, (u64) 0xffffffffffffffff);

            //網(wǎng)卡 I/O,memory資源的啟始地址

            tg3reg_base = pci_resource_start(pdev, 0);

            //網(wǎng)卡I/O,memory資源的大小

            tg3reg_len = pci_resource_len(pdev, 0);

            //分配并設置網(wǎng)絡設備

            dev = alloc_etherdev(sizeof(*tp));

            //申明為內(nèi)核設備模塊

            SET_MODULE_OWNER(dev);

            //初始化私有結(jié)構(gòu)中的各成員值

            tp = dev->priv;

            tp->pdev = pdev;

            tp->dev = dev;

            ……

            //鎖的初始化

            spin_lock_init(tp->lock);

            //映射I/O,memory地址到私有域中的寄存器結(jié)構(gòu)

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

            dev->irq = pdev->irq;

            //網(wǎng)絡設備回調(diào)函數(shù)賦值

            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;

            //網(wǎng)卡的MAC地址賦值dev->addr

            tg3_get_device_address(tp);

            //注冊網(wǎng)絡設備

            register_netdev(dev);

            //把網(wǎng)絡設備指針地址放入PCI設備中的設備指針中

            pci_set_drvdata(pdev, dev);

            }

            3.注銷網(wǎng)絡設備

            static void __devexit tg3_remove_one(struct pci_dev *pdev)

            {

            struct net_device *dev = pci_get_drvdata(pdev);

            //注銷網(wǎng)絡設備

            unregister_netdev(dev);

            //取消地址映射

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

            //釋放網(wǎng)絡設備

            kfree(dev);

            //釋放PCI資源

            pci_release_regions(pdev);

            //停用PCI設備

            pci_disable_device(pdev);

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

            pci_set_drvdata(pdev, NULL);

            }

            4.打開網(wǎng)絡設備

            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是向系統(tǒng)登記的中斷處理函數(shù)。這是一個回調(diào)函數(shù),中斷發(fā)生時,系統(tǒng)調(diào)用這個函數(shù),傳入的參數(shù)包括硬件中斷號,device id,寄存器值。dev_id就是下面的request_irq時傳遞給系統(tǒng)的參數(shù)dev_id。irqflags是中斷處理的一些屬性。比較重要的有SA_INTERRUPT,標明中斷處理程序是快速處理程序(設置SA_INTERRUPT)還是慢速處理程序(不設置SA_INTERRUPT)。快速處理程序被調(diào)用時屏蔽所有中斷。慢速處理程序不屏蔽。還有一個SA_SHIRQ屬性,設置了以后運行多個設備共享中斷。dev_id在中斷共享時會用到。一般設置為這個設備的device結(jié)構(gòu)本身或者NULL。中斷處理程序可以用dev_id找到相應的控制這個中斷的設備,或者用rq2dev_map找到中斷對應的設備。*/

            //初始化硬件

            tg3_init_hw(tp);

            //初始化收包和發(fā)包的緩沖區(qū)

            tg3_init_rings(tp);

          linux操作系統(tǒng)文章專題:linux操作系統(tǒng)詳解(linux不再難懂)

          上一頁 1 2 3 下一頁

          評論


          相關推薦

          技術專區(qū)

          關閉
          看屁屁www成人影院,亚洲人妻成人图片,亚洲精品成人午夜在线,日韩在线 欧美成人 (function(){ var bp = document.createElement('script'); var curProtocol = window.location.protocol.split(':')[0]; if (curProtocol === 'https') { bp.src = 'https://zz.bdstatic.com/linksubmit/push.js'; } else { bp.src = 'http://push.zhanzhang.baidu.com/push.js'; } var s = document.getElementsByTagName("script")[0]; s.parentNode.insertBefore(bp, s); })();