【Linux开发】linux设备驱动归纳总结(五):3.操作硬件——IO静态映射
xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx
有时候会觉得,每次访问硬件都要先通过ioremap来获取虚拟地址,其实有没有一种一劳永逸的方法,只要一次的操作,以后就能通过这个地址来访问硬件。答案是“有”,这就是接下来要介绍的IO内存静态映射。
xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx
一、静态IO是怎么建立的
Io静态映射发生在内核启动的时候,接下来通过内核源代码来分析,如果你的开发板是mini2440或者时候mini2440的内核配置文件,可以跟着我同样修改。注意:我的开发板只是使用mini2440的配置文件,外围电路跟mini2440不一样。
注:以下代码在内核目录linux-2.6.29/arch/arm/mach-s3c2440/mach-mini2440.c。
静态映射的建立方法,是在内核启动的时候,读取struct map_desc结构体里面的成员:
/*arch/arm/include/asm/mach/map.h*/
14 struct map_desc {
15 unsigned long virtual; //存放以后需要操作的虚拟地址,由自己定义
16 unsigned long pfn; //需要操作的硬件的物理地址对应的页帧号,即物理地址右移12
17 unsigned long length; //需要映射的大小
18 unsigned int type; //类型
19 };
这里要说明两个成员:
1)物理地址的页帧号pfn:如果你了解linux的页式管理,那你应该知道,一个页的大小是4096B(2
<< 12),所以一个地址的31-12位是用来表示一个地址对应的页帧号。对应的,一个物理地址,只要右移12位就能得到对应的页帧号,也可以使用函数:
#define __phys_to_pfn(paddr) ((paddr) >> PAGE_SHIFT) //其实也是右移12位
2)类型type:有下面这些类型定义:
/*include/asm/mach/map.h*/
21 /* types 0-3 are defined in asm/io.h */
22 #define MT_UNCACHED
4
23 #define MT_CACHECLEAN 5
24 #define MT_MINICLEAN 6
25 #define MT_LOW_VECTORS 7
26 #define MT_HIGH_VECTORS 8
27 #define MT_MEMORY 9
28 #define MT_ROM 10
其中,MT_UNCACHED是我们常用的,表示该地址不放在缓冲区cached中。要知道,为了方便内存的访问,内核会将一些经常使用的内存数据放在cached中,但是这样在访问寄存器时就不行了,如果寄存器改变了,内核读取数据是从cached中读取数据,而不在寄存器读取,这样的做法是不合理的。
首先,我们需要往这个结构体中填充我们需要访问的地址。
本来这个结构体是空的。
/*arch/arm/mach-s3c2440/mach-mini2440.c*/
45 static struct map_desc mini2440_iodesc[] __initdata = {
46 };
修改成:
45 static struct map_desc mini2440_iodesc[] __initdata = {
46 {
47 .virtual = 0xeeee0000,
48 .pfn = __phys_to_pfn(0x56000000), //0x56000
49 .length = SZ_4K, //这里我直接映射一页
50 .type = MT_UNCACHED
51 },
52 };
填充结构体后,我们再看看启动时通过调用什么函数:
/*linux-2.6.29/arch/arm/mach-s3c2440/mach-mini2440.c*/
264 static void __init mini2440_map_io(void)
265 { //就是这个函数,将我刚才修改的结构体的成员进行静态映射
266 s3c24xx_init_io(mini2440_iodesc, ARRAY_SIZE(mini2440_iodesc));
267 s3c24xx_init_clocks(12000000);
268 s3c24xx_init_uarts(mini2440_uartcfgs, ARRAY_SIZE(mini2440_uartcfgs));
269 }
这个函数里面有一个重要的函数——iotable_init(),其实大部分的工作都由这个函数来完成,实现静态映射。
既然修改了内核,就需要重新编译内核:
make bzImage
通过上面这几步,我们就实现了这样的一个操作,可以通过虚拟地址0xeeee0000来访问一页的物理地址。既然知道了虚拟地址和物理地址之间的关系,就不需要再用ioremap了。
为了更好的规范,我们使用一个头文件来定义寄存器的访问地址,方便编程时使用:
/*5th_mm_3/1st/test_map_io.h*/
1 #ifndef _TEST_H
2 #define _TEST_H
3
4 typedef volatile unsigned long * s3c_reg_t;
5
6 #define S3C2440_VA 0xeeee0000 //我们已经知道静态映射的虚拟地址
7
8 #define S3C2440_BASE(x) (S3C2440_VA + (x))
9 #define S3C2440_GPEBASE S3C2440_BASE(0x40)
10 #define S3C2440_GPECON S3C2440_BASE(0x40) //这就是我们要操作的寄存器
11 #define S3C2440_GPEDAT S3C2440_BASE(0x44)
12 #define S3C2440_GPEUP S3C2440_BASE(0x48)
13
14
15 #endif /* _TEST_H */
然后再修改一下前一节的2nd函数,去掉ioremap部分:
1 #include
2 #include
3
4 #include
5 #include "test_map_io.h"
6
7 s3c_reg_t *GPECON,
*GPEDAT, *GPEUP;
8 unsigned long reg;
9
10 void led_device_init(void)
11 {
12 GPECON = (s3c_reg_t
*)S3C2440_GPECON;
13 GPEDAT = (s3c_reg_t
*)S3C2440_GPEDAT;
14 GPEUP = (s3c_reg_t
*)S3C2440_GPEUP;
15 }
16
17 void led_configure(void)
18 {
19 reg = ioread32(GPECON);
20 reg &= ~(3 << 24);
21 reg |= (1 << 24);
22 iowrite32(reg, GPECON);
23
24 reg = ioread32(GPEUP);
25 reg &= ~(3 << 12);
26 iowrite32(reg, GPEUP);
27 }
28
29 void led_on(void)
30 {
31 reg = ioread32(GPEDAT);
32 reg &= ~(1 << 12);
33 iowrite32(reg, GPEDAT);
34 }
35
36 void led_off(void)
37 {
38 reg = ioread32(GPEDAT);
39 reg |= (1 << 12);
40 iowrite32(reg, GPEDAT);
41 }
42
43 static int __init test_init(void) //模块初始化函数
44 {
45 led_device_init();
46 led_configure();
47 led_on();
48 printk("hello led!\n");
49 return 0;
50 }
51
52 static void __exit test_exit(void) //模块卸载函数
53 {
54 led_off();
55 printk("bye\n");
56 }
57
58 module_init(test_init);
59 module_exit(test_exit);
60
61 MODULE_LICENSE("GPL");
62 MODULE_AUTHOR("xoao bai");
63 MODULE_VERSION("v0.1");
除了红笔部分,和删除的ioremap相关函数,其他部分都没有改动,效果还是一样,加载灯亮,卸载灯灭。
xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx
二、总结
这节介绍的内容确实是少:
1)内核中使用什么数据结构来管理静态映射IO。
2)内核时候什么函数在启动过程实现静态IO映射。
3)如何编写驱动函数来使用静态映射IO。
xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx
源代码:
5th_mm_3.rar
【Linux开发】linux设备驱动归纳总结(五):3.操作硬件——IO静态映射的更多相关文章
- linux设备驱动归纳总结(五):3.操作硬件——IO静态映射【转】
本文转载自:http://blog.chinaunix.net/uid-25014876-id-83299.html linux设备驱动归纳总结(五):3.操作硬件——IO静态映射 xxxxxxxxx ...
- linux设备驱动归纳总结
前言: (总结已经基本写完,这段时间我会从新排版和修正.错误总会有的,望能指正!) 前段时间学习了嵌入式驱动,趁着没开始找工作,这段时间我会每天抽出时间来复习. 我的总结是根据学习时的笔记(李杨老师授 ...
- 【Linux】linux设备驱动归纳总结
前言: (总结已经基本写完,这段时间我会从新排版和修正.错误总会有的,望能指正!) 前段时间学习了嵌入式驱动,趁着没开始找工作,这段时间我会每天抽出时间来复习. 我的总结是根据学习时的笔记(李杨老师授 ...
- 【Linux开发】linux设备驱动归纳总结(五):1.在内核空间分配内存
linux设备驱动归纳总结(五):1.在内核空间分配内存 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx ...
- 【Linux开发】linux设备驱动归纳总结(五):2.操作硬件——IO内存
linux设备驱动归纳总结(五):2.操作硬件--IO内存 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx ...
- 【Linux开发】linux设备驱动归纳总结(五):4.写个简单的LED驱动
linux设备驱动归纳总结(五):4.写个简单的LED驱动 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx ...
- linux设备驱动归纳总结(五):1.在内核空间分配内存【转】
本文转载自:http://blog.chinaunix.net/uid-25014876-id-79134.html linux设备驱动归纳总结(五):1.在内核空间分配内存 xxxxxxxxxxxx ...
- 【Linux开发】linux设备驱动归纳总结(六):1.中断的实现
linux设备驱动归纳总结(六):1.中断的实现 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx ...
- 【Linux开发】linux设备驱动归纳总结(六):2.分享中断号
linux设备驱动归纳总结(六):2.分享中断号 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx ...
随机推荐
- 51nod 1172 Partial Sums V2
题目 给出一个数组A,经过一次处理,生成一个数组S,数组S中的每个值相当于数组A的累加,比如:A = {1 3 5 6} => S = {1 4 9 15}.如果对生成的数组S再进行一次累加操作 ...
- java——适配器模式、策略模式
适配器模式: https://www.cnblogs.com/honger/p/5970283.html 策略模式: https://www.jianshu.com/p/3bcf55cf83d3
- koa2+redis+jwt token验证,简单注册登录
首先新建文件夹命名koa-server,npm init,相关包的安装就不说了,这是我的package.json 新建index.js文件,编码如下,config全局配置不用管,redis是一个简单的 ...
- Python之concurrent.futures模块的使用
concurrent.futures的作用: 管理并发任务池.concurrent.futures模块提供了使用工作线程或进程池运行任务的接口.线程和进程池API都是一样,所以应用只做最小 ...
- 20190716NOIP模拟赛T2 通讯(tarjan缩点+贪心)
题目描述 “这一切都是命运石之门的选择.” 试图研制时间机器的机关SERN截获了中二科学家伦太郎发往过去的一条短 信,并由此得知了伦太郎制作出了电话微波炉(仮). 为了掌握时间机器的技术,SERN总部 ...
- 灰度图像--图像分割 Sobel算子,Prewitt算子和Scharr算子平滑能力比较
学习DIP第47天 转载请标明本文出处:http://blog.csdn.net/tonyshengtan ,出于尊重文章作者的劳动,转载请标明出处!文章代码已托管,欢迎共同开发: https://g ...
- dup和dup2函数简单使用
dup函数 头文件和函数原型: #include <unistd.h> int dup(int oldfd); dup函数是用来打开一个新的文件描述符,指向和oldfd同一个文件,共享文件 ...
- Hbase Bulkload
前言 Apache HBase 是目前大数据系统中应用最为广泛的分布式数据库之一.我们经常面临向 HBase 中导入大量数据的情景,通常会选择使用标准的客户端 API 对 HBase 进行直接的操作, ...
- 系统芯片 SoC
SoC的定义多种多样,由于其内涵丰富.应用范围广,很难给出准确定义.一般说来, SoC称为系统级芯片,也有称片上系统,意指它是一个产品,是一个有专用目标的集成电路,其中包含完整系统并有嵌入软件的全部内 ...
- (十一)C语言之选择结构