Linux内核驱动--硬件访问I/O【原创】

寄存器与内存

寄存器与内存的区别在哪里呢?

寄存器和RAM的主要不同在于寄存器操作有副作用(side effect或边际效果):

读取某个地址时可能导致该地址内容发生变化,比如很多设备的中断状态寄存器只要一读取,便自动清零。

内存与IO

在X86处理器中存在IO空间的概念,IO空间是相对内存空间而言的,他们是彼此独立的地址空间,在32位的x86系统中,IO空间大小只有64K,内存却有4G

X86          支持内存空间、IO空间

ARM                 只支持内存空间

MIPS        只支持内存空间

PowerPC  只支持内存空间

IO端口:

当一个寄存器或内存位于IO空间时,称其为IO端口。

IO内存:

当一个寄存器或内存位于内存空间时,称其为IO内存。

操作I/O端口

对I/O端口的操作需按如下步骤完成:

1、  申请

2、  访问

3、  释放

申请I/O端口

内核提供了一套函数来允许驱动申请他需要的I/O端口,其中核心的函数是:

struct resource *request_region(unsigned long first, unsigned long n, const char *name)

这个函数告诉内核,你要使用从first开始的n个端口,name参数是设备的名字。如果申请成功,返回非NULL,如果申请失败,返回NULL.

系统中端口的分配情况记录在/proc/ioports中. cat  /proc/ioports,如果不能分配需要的端口,可以来这里看看谁在使用。

访问I/O端口

I/O端口可分为8位,16,32位端口,Linux内核头文件(体系依赖的头文件<asm/io.h>)定义了下列内联函数来访问I/O端口:

unsigned inb(unsigned port)           读字节端口(8位宽)

void outb(unsigned char byte, unsigned port)     写字节端口(8位宽)

unsigned inw(unsigned port)

void outw(unsigned short word, unsigned port)   存取16位端口

unsigned inl(unsigned port)

void outl(unsigned long word, unsigned port)       存取32位端口

释放I/O端口

当用完一组I/O端口(通常在驱动卸载的时候),用如下函数把它们返还给系统:

void release_region(unsigned long start, unsigned long n)

操作I/O内存

步骤:

1、  申请

2、  映射

3、  访问

4、  释放

申请I/O内存

核心函数为:

Struct resource *request_mem_region(unsigned long start, unsigned long len, char *name)

这个函数申请一个从start开始, 长度为len 字节的内存区。如果成功,返回非NULL,

否则返回NULL, 所有已经在使用的I/O内存在 /proc/iomem中列出

映射I/O内存

在访问IO内存之前,必须进行物理地址到虚拟地址的映射,

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

访问IO内存

从IO内存读:

unsigned ioread8(void *addr)

unsigned ioread16(void *addr)

unsigned ioread32(void *addr)

写IO内存

void iowrite8(u8 value, void *addr)

void iowrite16(u16 value, void *addr)

void iowrite32(u32 value, void *addr)

老版本的I/O内存访问函数:

从I/O内存读,

unsigned readb(address)

unsigned readw(address)

unsigned readl(address)

写IO内存

unsigned writeb(unsigned value, address)

unsigned writew(unsigned value, address)

unsigned writel(unsigned value, address)

释放IO内存

IO内存不再需要使用时应当释放,步骤:

1、  void iounmap(void *addr)

2、  void release_mem_region(unsigned long start, unsigned long len)

欢迎交流

如有转载请注明出处

新浪博客:http://blog.sina.com.cn/u/2049150530
博客园:http://www.cnblogs.com/sky-heaven/
知乎:http://www.zhihu.com/people/zhang-bing-hua

时间: 2024-12-09 21:33:12

Linux内核驱动--硬件访问I/O【原创】的相关文章

Linux内核驱动--mmap设备方法【原创】

mmap系统调用(功能) void *mmap(void *addr, size_t len, int prot, int flags, int fd, off_t offset) 内存映射函数mmap , 负责把文件内容映射到进程的虚拟内存空间,通过对这段内存的读取和修改,来实现对文件的读取和修改,而不需要再调用read, write等操作. addr:        指定映射的起始地址,通常设为NULL, 由系统指定. len:          映射到内存的文件长度 prot:     

linux内核驱动模型

linux内核驱动模型,以2.6.32内核为例.(一边写一边看的,有点乱.) 1.以内核对象为基础.用kobject表示,相当于其它对象的基类,是构建linux驱动模型的关键.具有相同类型的内核对象构成内核对象集,用kset表示,内核对象集也包含自己的内核对象,从而组成层次化的结构.2.用sysfs文件系统导出到用户空间.内核中的所有内核对象组织成树状,以对象属性为叶子.通过sysfs文件系统,将用户空间对文件的读写操作转化为对内核对象属性的显示和保存方法.从而导出内核对象信息,并提供配置接口.

第02节:Linux 内核驱动中的指定初始化

2.1 什么是指定初始化 在标准 C 中,当我们定义并初始化一个数组时,常用方法如下: int a[10] = {0,1,2,3,4,5,6,7,8}; 按照这种固定的顺序,我们可以依次给 a[0] 和 a[8] 赋值.因为没有对 a[9] 赋值,所以编译器会将 a[9] 默认设置为0.当数组长度比较小时,使用这种方式初始化比较方便.当数组比较大,而且数组里的非零元素并不连续时,这时候再按照固定顺序初始化就比较麻烦了. 比如,我们定义一个数组 b[100],其中 b[10].b[30] 需要初始

Linux内核驱动编程

Linux内核驱动编程 2015-02-12 驱动程序基础的东西这儿就不罗嗦了,百度上有更好的资料,此处我们只是注重实际用处. 下面我们开始写程序: 一.初步helloword程序 首先是来一个简单的hello. hello.c代码: 1 /****************************** 2 3 the first program 4 5 Hello World! 6 7 ******************************/ 8 9 #include <linux/mod

linux 内核驱动--Platform Device和Platform_driver注册过程

linux 内核驱动--Platform Device和Platform_driver注册过程 从 Linux 2.6 起引入了一套新的驱动管理和注册机制 :Platform_device 和 Platform_driver . Linux 中大部分的设备驱动,都可以使用这套机制 , 设备用 Platform_device 表示,驱动用 Platform_driver 进行注册. Linux platform driver 机制和传统的 device driver 机制 ( 通过 driver_

Unix/Linux环境C编程入门教程(12) openSUSECCPP以及Linux内核驱动开发环境搭建

1. openSUSE是一款优秀的linux. 2.选择默认虚拟机 3.选择稍后安装操作系统 4.选择linux  opensuse 5. 选择默认虚拟机名称 6.设置处理器为双核. 7.内存设置为2G 8. 选择网络地址转换 9.设置IO控制器 10. 选择默认磁盘类型 11.创建一个新的虚拟磁盘 12.设置磁盘大小 13.选择路径保存虚拟磁盘 14. 完成虚拟机创建 15.设置虚拟机 16.选择opensuse镜像 17.开启虚拟机 18.虚拟机启动 19.安装opensuse 20.安装程

Linux内核驱动注册方式泛谈

Linux驱动注册有多种方式,通常是以内核提供的表征数据结构封装后按照内核子系统提供的接口函数进行注册,还有一些是比较复杂的以链表方式进行维护.以下对几种驱动注册方式进行介绍: 一.子系统有专门的驱动注册函数: 例如RTC子系统,提供rtc_device_register注册接口函数. 例如: rtc_device_register(client->name,&client->dev, &rx8025_rtc_ops, THIS_MODULE); static struct r

Unix/Linux环境C编程新手教程(12) openSUSECCPP以及Linux内核驱动开发环境搭建

1. openSUSE是一款优秀的linux. watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvaXRjYXN0Y3Bw/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/SouthEast" /> 2.选择默认虚拟机 3.选择稍后安装操作系统 4.选择linux  opensuse 5. 选择默认虚拟机名称 6.设置处理器为双核. watermark/2/text/a

你为什么看不懂Linux内核驱动源码?

学习嵌入式Linux驱动开发,最核心的技能就是能够编写Linux内核驱动.深入理解Linux内核.而做到这一步的基础,就是你要看得懂Linux内核源码,了解其基本的框架和具体实现,了解其内核API的使用方法,然后才能根据自己的需求写出高质量的内核驱动程序. 说易行难,很多新人.甚至工作1-2年的开发者刚接触Linux内核时,别说写了,看内核代码可能都是一脸懵逼:明明是C语言,但是就是看不懂是什么意思,除了根据函数名.函数参数.函数的返回值以及注释,了解整个函数的基本功能外,一旦分析其细节,你会发