日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

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

linux

linux如何加载并口驱动程序,Linux下并口的访问★★★★★

發(fā)布時(shí)間:2024/1/1 linux 32 豆豆
生活随笔 收集整理的這篇文章主要介紹了 linux如何加载并口驱动程序,Linux下并口的访问★★★★★ 小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.

Linux I/O port programming mini-HOWTO

Author: Riku Saikkonen v3.0, 2000-12-13

This HOWTO document describes programming hardware I/O ports and waiting for small periods of time in user-mode Linux programs running on the Intel x86 architecture.

===========================================

linux并口驅(qū)動(dòng)代碼有問題,幫忙看看。

重新啟動(dòng)系統(tǒng)以后加載概模塊,提示說0x378端口已經(jīng)被占用,所以我卸載了lp 和 parport_pc模塊,可以加載成功,可是我卸載以后再重新加載不成功,又說資源被占用。可能是卸載模塊時(shí)沒有釋放資源,可是代碼里面明明有這一部分代碼阿???郁悶中。。。。。。。

卸載模塊時(shí)提示:

Trying to free nonexistent resource <00000378-0000037f>

代碼如下:

#ifndef __KERNEL__

# define __KERNEL__

#endif

#ifndef MODULE

# define MODULE

#endif

//#include

#include

#include

#include

#include

#include

#include /* printk() */

#include /* everything... */

#include /* error codes */

#include /* udelay */

//#include

#include

#include

#include

#include

#include

#include

#ifdef MODULE_LICENSE

MODULE_LICENSE ("GPL");

#endif

static unsigned long base = 0x378;

unsigned long short_base = 0x378;

static unsigned long major = 0;

static int portnum = 8;

int

short_open (struct inode *inode, struct file *filp)

{

MOD_INC_USE_COUNT;

return 0;

}

int

short_release (struct inode *inode, struct file *filp)

{

MOD_DEC_USE_COUNT;

return 0;

}

ssize_t

do_short_write (struct inode * inode, struct file * filp, const char *buf,

size_t count, loff_t * f_pos)

{

int retval = count;

unsigned long address = short_base + 2;

unsigned char *kbuf = kmalloc (count, GFP_KERNEL), *ptr;

if (!kbuf)

return -ENOMEM;

if (copy_from_user (kbuf, buf, count))

return -EFAULT;

ptr = kbuf;

while (count--)

{

outb (*(ptr++), address);

wmb ();

}

kfree (kbuf);

return retval;

}

ssize_t

do_short_read (struct inode * inode,

struct file * filp, char *buf, size_t count, loff_t * f_pos)

{

int retval = count;

unsigned long address = short_base + 2;

unsigned char *kbuf = kmalloc (count, GFP_KERNEL), *ptr;

if (!kbuf)

return -ENOMEM;

ptr = kbuf;

while (count--)

{

*(ptr++) = inb (address);

rmb ();

}

if ((retval > 0) && copy_to_user (buf, kbuf, retval))

retval = -EFAULT;

kfree (kbuf);

return retval;

}

ssize_t

short_read (struct file * filp, char *buf, size_t count, loff_t * f_pos)

{

return do_short_read (filp->f_dentry->d_inode, filp, buf, count, f_pos);

}

ssize_t

short_write (struct file * filp, const char *buf, size_t count, loff_t * f_pos)

{

return do_short_write (filp->f_dentry->d_inode, filp, buf, count, f_pos);

}

struct file_operations short_fops = {

read:short_read,

write:short_write,

open:short_open,

release:short_release,

};

int

init_module (void)

{

int portnum = 1;

int result;

SET_MODULE_OWNER (&short_fops);

result = check_region (base, portnum);

if (result)

{

printk (KERN_INFO "short: can't get I/O port address 0x%lx ",

short_base);

return result;

}

request_region (base, portnum, "short");

result = register_chrdev (major, "short", &short_fops);

if (result < 0)

{

printk (KERN_INFO "short: can't get major number ");

release_region (short_base, portnum);

return result;

}

if (major == 0)

major = result;

return 0;

}

void

cleanup_module (void)

{

printk ("major = 0x%lx ", major);

unregister_chrdev (major, "short");

release_region (base, portnum);

}

編譯:

gcc -D__KERNEL__ -O2 -Wall -I/usr/src/linux-$(shell uname -r)/include -c pell.c

------------------------

已經(jīng)找到原因,這是你的一個(gè)小小的筆誤,就是你可以查看一下你在

init_module中有這么一句話

int portnum = 1;

而你在前面定義了一個(gè)全局變量

static int portnum = 8;

這樣在編譯器處理的時(shí)候就會(huì)把portnum寫成1,這樣當(dāng)你用release_region釋放資源的時(shí)候就會(huì)發(fā)現(xiàn)沒有 合適的資源要釋放

這是因?yàn)槟氵@時(shí)要釋放的是8個(gè)字節(jié)長(zhǎng)的resource,而你前面申請(qǐng)的只有1個(gè)byte長(zhǎng)的resource ,這樣就會(huì)造成資源泄漏.

謝謝你提供了源代碼,不然,真的沒有人能夠找出這其中的毛病,

------------------開放就意味著找出更多的不足,得到更大的改進(jìn).

============================================

兩種方式

一? 驅(qū)動(dòng)程序執(zhí)行方式

1.申請(qǐng)I/O端口

A 直接端口方式

check_region

request_region

B內(nèi)存映射方式

check_mem_region

request_mem_retion

然后隊(duì)端口地址映射

ioremap

2.注冊(cè)驅(qū)動(dòng)

register_chrdev? 申請(qǐng)主設(shè)備號(hào),注冊(cè)驅(qū)動(dòng)名,相關(guān)的操作.

3.探測(cè)中斷

A 內(nèi)核探測(cè)

B 定制探測(cè)

C 直接根據(jù)I/O地址,分配相應(yīng)的IRQ號(hào)

4.安裝中斷相應(yīng)處理函數(shù)

request_irq(irq,(*)Handler()...)

A 共享中斷的處理函數(shù)

端口有中斷到,則判斷時(shí)不時(shí)本端口的中斷.若是.則填充緩沖區(qū).同時(shí)wake_up_interruptible,喚起等待序列.

在read操作中的隊(duì)列由signal_pending喚醒,拷貝數(shù)據(jù)到用戶空間.

B 下半部中斷的處理函數(shù)

通過queue_task,執(zhí)行隊(duì)列處理函數(shù)喚起中斷.

C? 小任務(wù)中斷處理函數(shù)

通過tasklet_schedule執(zhí)行小任務(wù)處理函數(shù)喚起中斷.

二?? 應(yīng)用程序執(zhí)行方式.通過ioperm命令,例如,ioperm ( BASE, range , 1),調(diào)用內(nèi)核,得到 I/O 地址空間的使用權(quán);

通過一個(gè)發(fā)送請(qǐng)求指令,例如,outb(1, BASE ),

等待足夠的時(shí)間讓咖啡煮好,讓時(shí)間參數(shù)在命令行中被讀取是一件很好的事情

然后發(fā)送 out(0, BASE) 指令關(guān)掉咖啡機(jī)

在結(jié)束之前還應(yīng)歸還并口 I/O 地址的使用權(quán),ioperm(BASE,range,0) .???? 程序:/* coffee.c */

#include /* linux-specific */

#ifdef __GLIBC__

#? include

#endif

int main(int argc, char **argv)

{

setuid(0); /* if we're setuid, force it on */

if(ioperm(0x378,1,1))

printf("error,we can't? ioperm our ox378 port\n");

outb(0xff,0x378);

sleep(5);

outb(0,0x378);

if(ioperm(0x378,1,0))

printf("error,we can't? ioperm our ox378 port\n");

exit(1);

}

三?? 在驅(qū)動(dòng)程序中如何取得被其它驅(qū)動(dòng)程序使用的并口

首先?? cat?? /proc/ioports? 看端口地址是分配給誰了.

然后看該名字的 ls -l? /dev/port_name

找到主設(shè)備號(hào)和次設(shè)備號(hào)

然后? cat?? /proc/devices 看主設(shè)備好對(duì)應(yīng)的驅(qū)動(dòng)

然后rmmod 該驅(qū)動(dòng).

如果是編譯到了內(nèi)核的驅(qū)動(dòng)則只需在/lib/module/`uname?? -r`/kernel/drivers/ 刪除該名字的驅(qū)動(dòng),

則重起后,該驅(qū)動(dòng)就不會(huì)暫用該端口了,但這個(gè)驅(qū)動(dòng)的名字不一定在/proc/devices 中出現(xiàn)

=============================================

Linux并口網(wǎng)絡(luò)解決方案(轉(zhuǎn)貼)

2003-04-10 nesta1

大家知道,在DOS環(huán)境下,我們可以用并口或串口將兩臺(tái)PC連接起來,一臺(tái)充當(dāng)服務(wù)器,另一臺(tái)充當(dāng)客戶,但充當(dāng)服務(wù)器的機(jī)器不能做其它操作,只能為 Client服務(wù)。雖然在方便上和速度均不如網(wǎng)卡,但它提供了一個(gè)“窮人”的解決方案。如果僅拷貝少量數(shù)據(jù),它還是可以滿足一般人的需求。并口的速度要遠(yuǎn)遠(yuǎn)比串口快。

在Linux內(nèi)核中,網(wǎng)絡(luò)設(shè)備中有一個(gè)叫PLIP (Parallel Line Internet Protocol). 它提供了并口的網(wǎng)絡(luò)支持,并將并口映射成網(wǎng)絡(luò)設(shè)備。它支持標(biāo)準(zhǔn)并口,擴(kuò)展并口的支持。傳送速度依賴于并口線的質(zhì)量和機(jī)器的配置。

下面,我將系統(tǒng)的配置作簡(jiǎn)要介紹。

1. 在內(nèi)核中支持PLIP

cd /usr/src/linux

make menuconfig

# select Network device support

select PLIP as modules

2. 編譯內(nèi)核

cd /usr/src/linux

make dep; make bzImage; make modules; make modules_install;

3. 將新內(nèi)核配置到Lilo中去。

4. 重新啟動(dòng)

5. 打開Ipforward.

echo 1 > /proc/sys/net/ipv4/ip_forward

6. 運(yùn)行網(wǎng)絡(luò)配置工具,配置PLIP

turbonetcfg

# add new interface

# select PLIP

# add the ipaddress and mask to that

# save & exit

7. 啟動(dòng)PLIP

modprobe plip

# if it does not work

# echo 7 > /proc/parport/0/irq

# modprobe plip

ifup plip0

8. 配置網(wǎng)關(guān)。

# eg. Machine A:

# PLIP0 -- 10.0.0.1

#B -- gateway

# PLIP0 -- 10.0.0.2

# eth0 -- 172.16.69.12

# in B Machine, we setup ipchains

/sbin/ipchains -A forward -j MASQ -s 10.0.0.0/255.255.255.0 -d 0.0.0.0/0

通過我們的試驗(yàn),使用WWW, TELNET 與普通網(wǎng)卡沒有區(qū)別,但當(dāng)使用FTP時(shí),速度稍慢,大約35K/s

左右,并且在FTP的同時(shí),我們也發(fā)現(xiàn)系統(tǒng)偶爾出現(xiàn)Timeout, 并且當(dāng)你作其它事情時(shí),感到系統(tǒng)很慢。說

明PLIP的驅(qū)動(dòng)程序還需要改進(jìn)。

參考資料:/usr/src/linux/Documentation/networking/PLIP.txt

===============================================

9.3.?一個(gè) I/O 端口例子

我們用來展示一個(gè)設(shè)備驅(qū)動(dòng)內(nèi)的端口 I/O 的例子代碼, 操作通用的數(shù)字 I/O 端口; 這樣的端口在大部分計(jì)算機(jī)系統(tǒng)中找到.

一個(gè)數(shù)字 I/O 端口, 在它的大部分的普通的化身中, 是一個(gè)字節(jié)寬的 I/O 位置, 或者內(nèi)存映射的或者端口映射的. 當(dāng)你寫一個(gè)值到一個(gè)輸出位置, 在輸出管腳上見到的電信號(hào)根據(jù)寫入的單個(gè)位而改變. 當(dāng)你從一個(gè)輸入位置讀取一個(gè)值, 輸入管腳上所見的當(dāng)前邏輯電平作為單個(gè)位的值被返回.

這樣的 I/O 端口的實(shí)際實(shí)現(xiàn)和軟件接口各個(gè)系統(tǒng)不同. 大部分時(shí)間, I/O 管腳由 2 個(gè) I/O 位置控制: 一個(gè)允許選擇使用那些位作為輸入, 哪些位作為輸出, 以及一個(gè)可以實(shí)際讀或?qū)戇壿嬰娖降? 有時(shí), 但是, 事情可能更簡(jiǎn)單, 并且這些位是硬連線為輸入或輸出(但是, 在這個(gè)情況下, 它們不再是所謂的"通用 I/O"); 在所有個(gè)人計(jì)算機(jī)上出現(xiàn)的并口是這樣一個(gè)非通用 I/O 端口. 任一方式, I/O 管腳對(duì)我們馬上介紹的例子代碼是可用的.

9.3.1.?并口縱覽

因?yàn)槲覀兤谕蟛糠肿x者以所謂的"個(gè)人計(jì)算機(jī)"的形式使用一個(gè) x86 平臺(tái), 我們覺得值得解釋一下 PC 并口如何設(shè)計(jì)的. 并口是在個(gè)人計(jì)算機(jī)上運(yùn)行數(shù)字 I/O 例子代碼的外設(shè)接口選擇. 盡管大部分讀者可能有并口規(guī)范用, 為你的方便, 我們?cè)谶@里總結(jié)一下它們.

并口, 在它的最小配置中 ( 我們?yōu)g覽一下 ECP 和 EPP 模式) 由 3 個(gè) 8-位端口組成. PC 標(biāo)準(zhǔn)在 0x378 開始第一個(gè)并口的 I/O 端口并且第 2 個(gè)在 0x278. 第一個(gè)端口是一個(gè)雙向數(shù)據(jù)寄存器; 它直接連接到物理連接器的管腳 2 - 9. 第 2 個(gè)端口是一個(gè)只讀狀態(tài)寄存器; 當(dāng)并口為打印機(jī)使用, 這個(gè)寄存器報(bào)告打印機(jī)狀態(tài)的幾個(gè)方面, 例如正在線, 缺紙, 或者忙. 第 3 個(gè)端口是一個(gè)只出控制寄存器, 它, 在其他東西中, 控制是否中斷使能.

并口通訊中使用的信號(hào)電平是標(biāo)準(zhǔn)的 TTL 電平: 0 和 5 伏特, 邏輯門限在大概 1.2 伏特. 你可依靠端口至少符合標(biāo)準(zhǔn) TTL LS 電流規(guī)格, 盡管大部分現(xiàn)代并口在電流和電壓額定值都工作的好.

并口連接器和計(jì)算機(jī)內(nèi)部電路不隔離, 當(dāng)你想直接連接邏輯門到這個(gè)端口是有用的. 但是你不得不小心地正確連接線; 并口電路當(dāng)你使用你自己的定制電路時(shí)容易損壞, 除非你給你的電路增加絕緣. 你可以選擇使用插座并口如果你害怕會(huì)損壞你的主板.

位的規(guī)范在圖 并口的管腳 中概述. 你可以存取 12 個(gè)輸出位和 5 個(gè)輸入位, 有些是在它們地信號(hào)路徑上邏輯地翻轉(zhuǎn)了. 唯一的沒有關(guān)聯(lián)信號(hào)管腳的位是端口 2 的位 4 (0x10), 它使能來自并口的中斷. 我們使用這個(gè)位作為我們的在第 10 章中的中斷處理的實(shí)現(xiàn)的一部分.

圖?9.1.?并口的管腳

9.3.2.?一個(gè)例子驅(qū)動(dòng)

我們介紹的驅(qū)動(dòng)稱為 short (Simple Hardware Operations and Raw Tests). 所有它做的是讀和寫幾個(gè) 8-位 端口, 從你在加載時(shí)選擇的開始. 缺省地, 它使用分配給 PC 并口的端口范圍. 每個(gè)設(shè)備節(jié)點(diǎn)(有一個(gè)獨(dú)特的次編號(hào))存取一個(gè)不同的端口. short 驅(qū)動(dòng)不做任何有用的事情; 它只是隔離來作為操作端口的單個(gè)指令給外部使用. 如果你習(xí)慣端口 I/O, 你可以使用 short 來熟悉它; 你能夠測(cè)量它花費(fèi)來通過端口傳送數(shù)據(jù)的時(shí)間或者其他游戲的時(shí)間.

為 short 在你的系統(tǒng)上運(yùn)行, 必須有存取底層硬件設(shè)備的自由(缺省地, 并口); 因此, 不能有其他驅(qū)動(dòng)已經(jīng)分配了它. 大部分現(xiàn)代發(fā)布設(shè)置并口驅(qū)動(dòng)作為只在需要時(shí)加載的模塊, 因此對(duì) I/O 地址的競(jìng)爭(zhēng)常常不是個(gè)問題. 如果, 但是, 你從 short 得到一個(gè)"無法獲得 I/O 地址" 錯(cuò)誤(在控制臺(tái)上或者在系統(tǒng) log 文件), 一些其他的驅(qū)動(dòng)可能已經(jīng)獲得這個(gè)端口. 一個(gè)快速瀏覽 /proc/ioports 常常告訴你哪個(gè)驅(qū)動(dòng)在搗亂. 同樣的告誡應(yīng)用于另外 I/O 設(shè)備如果你沒有在使用并口.

從現(xiàn)在開始, 我們只是用"并口"來簡(jiǎn)化討論. 但是, 你能夠設(shè)置基本的模塊參數(shù)在加載時(shí)來重定向 short 到其他 I/O 設(shè)備. 這個(gè)特性允許例子代碼在任何 Linux 平臺(tái)上運(yùn)行, 這里你對(duì)一個(gè)數(shù)字 I/O 接口有權(quán)限通過 outb 和 inb 存取( 盡管實(shí)際的硬件是內(nèi)存映射的, 除 x86 外的所有平臺(tái)). 后面, 在"使用 I/O 內(nèi)存"的一節(jié), 我們展示 short 如何用來使用通用的內(nèi)存映射數(shù)字 I/O.

為觀察在并口上發(fā)生了什么以及如果你有使用硬件的愛好, 你可以焊接盡管 LED 到輸出管腳. 每個(gè) LED 應(yīng)當(dāng)串連一個(gè) 1-K 電阻導(dǎo)向一個(gè)地引腳(除非, 當(dāng)然, 你的 LED 有內(nèi)嵌的電阻). 如果你連接一個(gè)輸出引腳到一個(gè)輸入管腳, 你會(huì)產(chǎn)生你自己的輸入能夠從輸入端口讀到.

注意, 你無法只連接一個(gè)打印機(jī)到并口并且看到數(shù)據(jù)發(fā)向 short. 這個(gè)驅(qū)動(dòng)實(shí)現(xiàn)簡(jiǎn)單的對(duì) I/O 端口的存取, 并且沒有進(jìn)行與打印機(jī)需要的來操作數(shù)據(jù)的握手; 在下一章, 我們展示了一個(gè)例子驅(qū)動(dòng)(稱為 shortprint ), 它能夠驅(qū)動(dòng)并口打印機(jī); 這個(gè)驅(qū)動(dòng)使用中斷, 但是, 因此我們還是不能到這一點(diǎn).

如果你要查看并口數(shù)據(jù)通過焊接 LED 到一個(gè) D-型 連接器, 我們建議你不要使用管腳 9 和管腳 10, 因?yàn)槲覀冎筮B接它們?cè)谝黄饋磉\(yùn)行第 10 章展示的例子代碼.

只考慮到 short, /dev/short0 寫到和讀自位于 I/O 基地址的 8-bit 端口( 0x378, 除非在加載時(shí)間改變). /dev/short1 寫到位于基址 + 1 的 8-位, 等等直到基址 + 7.

/dev/short0 進(jìn)行的實(shí)際輸出操作是基于使用 outb 的一個(gè)緊湊循環(huán). 一個(gè)內(nèi)存屏障指令用來保證輸出操作實(shí)際發(fā)生并且不被優(yōu)化掉:while (count--) {

outb(*(ptr++), port);

wmb();

}

你可以運(yùn)行下列命令來點(diǎn)亮你的 LED:echo -n "any string" > /dev/short0

每個(gè) LED 監(jiān)視一個(gè)單個(gè)的輸出端口位. 記住只有最后寫入的字符, 保持穩(wěn)定在輸出管腳上足夠長(zhǎng)時(shí)間你的眼睛能感覺到. 因此, 我們建議你阻止自動(dòng)插入一個(gè)結(jié)尾新行, 通過傳遞一個(gè) -n 選項(xiàng)給 echo.

讀是通過一個(gè)類似的函數(shù), 圍繞 inb 而不是 outb 建立的. 為了從并口讀"有意義的"值, 你需要某個(gè)硬件連接到連接器的輸入管腳來產(chǎn)生信號(hào). 如果沒有信號(hào), 你會(huì)讀到一個(gè)相同字節(jié)的無結(jié)尾的流. 如果你選擇從一個(gè)輸出端口讀取, 你極可能得到寫到端口的最后的值(這適用于并口和普通使用的其他數(shù)字 I/O 電路). 因此, 那些不喜歡拿出他們的烙鐵的人可以讀取當(dāng)前的輸出值在端口 0x378, 通過運(yùn)行這樣一個(gè)命令:dd if=/dev/short0 bs=1 count=1 | od -t x1

為演示所有 I/O 指令的使用, 每個(gè) short 設(shè)備有 3 個(gè)變形: /dev/short0 進(jìn)行剛剛展示的循環(huán), /dev/short0p 使用 outb_p 和 inb_p 代替"快速"函數(shù), 并且 /dev/short0s 使用字串指令. 有 8 個(gè)這樣的設(shè)備, 從 short0 到 short7. 盡管 PC 并口只有 3 個(gè)端口, 你可能需要它們更多如果使用不同的 I/O 設(shè)備來運(yùn)行你的測(cè)試.

short 驅(qū)動(dòng)進(jìn)行一個(gè)非常少的硬件控制, 但是足夠來展示如何使用 I/O 端口指令. 感興趣的讀者可能想看看 parpor 和 parport_pc 模塊的源碼, 來知道這個(gè)設(shè)備在真實(shí)生活中能有多復(fù)雜來支持一系列并口上的設(shè)備(打印機(jī), 磁帶備份, 網(wǎng)絡(luò)接口)

總結(jié)

以上是生活随笔為你收集整理的linux如何加载并口驱动程序,Linux下并口的访问★★★★★的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問題。

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