Linux I/O端口与I/O内存

本文介绍在Linux环境下如何通过直接使用IO端口操作函数和将IO端口重新映射的方法来访问IO端口,以及如何通过申请资源、映射物理地址到内核空间来访问IO内存。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一、IO端口访问 

1、直接使用IO端口操作函数

1)在设备打开或驱动模块被加载时申请IO端口区域,之后使用inb(),outb()等进行端口访问,最后在设备关闭或驱动被卸载时释放IO端口范围。流程如下:

   


2)struct resource *request_region(unsigned long first, unsigned long n, const char *name);
        如果这段I/O端口没有被占用,在我们的驱动程序中就可以使用它。在使用之前,必须向系统登记,以防止被其他程序占用。登记后,在/proc/ioports文件中可以  看到你登记的io口。
  参数1:io端口的基地址。
  参数2:io端口占用的范围。
  参数3:使用这段io地址的设备名。
  在对I/O口登记后,就可以放心地用inb(), outb()之类的函来访问了。

        request_region()用于内核为驱动“分配”端口,这里分配的意思是,记录该端口已经被某个进程使用,要是其它进程试图访问它,就会产生“忙”错误。所以目的在于实现资源的互斥访问。

        反之, 如果一个资源只被一个进程访问,不会导致资源的争用,这时request_region()是可选的。


3)void release_region(unsigned long from, unsigned long num)
        这个函数用来释放一块输入输出区域。


2、将IO端口重新映射

1)使其看起来像I/O内存,在设备打开或驱动模块被加载时,申请IO端口区域并使用ioport_map()映射到内存,之后使用IO内存的函数进行端口访问,最后,在设备关闭或驱动模块被卸载时释放IO端口并释放映射,流程如下:

   


2)void *ioport_unmap(void *addr);

        ioport_map仅仅是将port加上PIO_OFFSET(64k),而ioport_unmap则什么都不做。这样portio的64k空间就被映射到虚拟地址的64k~128k之间,而ioremap返回的虚拟地址则肯定在3G之上。这样portio和mmio的虚拟地址就被统一起来。



二、IO内存访问

1)首先调用request_mem_region()申请资源,接着将寄存器地址通过ioremap()映射到内核空间的虚拟地址,之后就可以Linux设备访问编程接口访问这些虚拟地址了,访问完成后,使用iounmap()对申请的虚拟地址进行释放,并释放release_mem_region()申请的IO内存资源。流程如下:

   


2)void *ioremap(unsigned long phys_addr,unsigned long size);

ioremap函数用来将I/O资源的物理地址映射到内核虚拟地址空间(3G-4G),但是由Ioremap返回的地址不应直接引用,而应该使用内核提供的ioread,iowrite函数

       

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

luckywang1103

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值