生死契阔,与子成说,执子之手,与子携老

一、ioremap

1.参考:

理解

(1)http://www.linuxidc.com/Linux/2011-04/34295.htm

代码过程

(1)http://wenku.baidu.com/link?url=ol3FJuVkhMrs80Q8wbGsFjTwclEMKZdN2fzsaqggSnGEPyqLuWOI0YDvXil0P2cp5UHQrd4Za7BWEuQGGN2NKrhbFYxxUqgFG-wMM5uFTN_

(2)

http://hi.baidu.com/zengzhaonong/item/a2dc5eb4f805fe9719469726

ioremap代码参数

(1)

http://baike.baidu.com/link?url=vfrIFDTnwR1aW_yDOkVLvnoQci4TwmD8jT28m9KSdTL5LEnyfG_nAB1Wcb9aozQSnjrJNMOu28qbL83SmdWxsK

(2)

http://blog.csdn.net/fuyjlu/article/details/5039782

(3)

http://www.hh010.com/os/html/47-2/2811.htm

2.实现

(1) Linux在io.h头文件中声明了函数ioremap(),用来将I/O内存资源的物理地 址映射到核心虚地址空间(3GB-4GB)中,原型如下:

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

  iounmap函数用于取消ioremap()所做的映射,原型如下:

void iounmap(void * addr);

  这两个函数都是实现在mm/ioremap.c文件中。

(2)

  在将I/O内存资源的物理地址映射成核心虚地址后,理论上讲我们就可以象读写RAM那样直接读写I/O内存资源了。为了保证驱动程序的跨平台的 可移植性,我们应该使用Linux中特定的函数来访问I/O内存资源,而不应该通过指向核心虚地址的指针来访问。如在x86平台上,读写I/O的函数如下 所示:

#define readb(addr) (*(volatile unsigned char *) __io_virt(addr))
#define readw(addr) (*(volatile unsigned short *) __io_virt(addr))
#define readl(addr) (*(volatile unsigned int *) __io_virt(addr))

#define writeb(b,addr) (*(volatile unsigned char *) __io_virt(addr) = (b))
#define writew(b,addr) (*(volatile unsigned short *) __io_virt(addr) = (b))
#define writel(b,addr) (*(volatile unsigned int *) __io_virt(addr) = (b))

#define memset_io(a,b,c) memset(__io_virt(a),(b),(c))
#define memcpy_fromio(a,b,c) memcpy((a),__io_virt(b),(c))
#define memcpy_toio(a,b,c) memcpy(__io_virt(a),(b),(c))

(3)

在内核驱动程序的初始化阶段,通过ioremap()将物理地址映射到内核虚拟空间;在驱动程序的mmap系统调用中,使用remap_page_range()将该块ROM映射到用户虚拟空间。这样内核空间和用户空间都能访问这段被映射后的虚拟地址。

remap_page_range函数的功能是构造用于映射一段物理地址的新页表,实现了内核空间与用户空间的映射,其原型如下:

int remap_page_range(vma_area_struct *vma, unsigned long from, unsigned long to, unsigned long size, pgprot_tprot);

(4)

在X86体系下的,CPU的物理地址和PCI总线地 址共用一个空间。linux内核将3G-4G的虚拟地址固定映射到了物理地址的0-1G的地方。但是如果外围设备上的地址高于1G,例如某块PCI卡分配 到了一个高于1G的地址,就需要调用ioremap来重新建立该物理地址(总线地址)和虚拟地址之间的映射。这个映射过程是这样的:在ioremap.c 文件的__ioremap函数中首先对将来映射的物理地址进行检查,也就是不能重新映射640K-1M地址(由于历史的原因,物理地址640k到1M空间 被保留给了显卡),普通的ram地也不能重新被映射。之后调用get_vm_area获得可用的虚拟地址,然后根这虚拟地址和欲映射的物理地址修改页表, 之后内核就可以用这个虚拟地址来访问映射的物理地址了。

二、buddy system

 

最新文章

  1. NIO及Reactor模式
  2. Java小知识--length,length(),size()方法详细介绍
  3. Centos下Apache使用Symlink访问外部目录出现403
  4. 最近读的javascript,一些文章
  5. git some cookies
  6. Thinkphp excel导入导出
  7. ASP.NET MVC性能优化工具 MiniProfiler
  8. js获取url中的参数方法
  9. 局域网内部署 Docker Registry
  10. string,char*,int 之间的转化
  11. 9个Console命令
  12. python全栈开发 * 03 基本数据类型 * 180601
  13. LeetCode——150. Evaluate Reverse Polish Notation
  14. Java SPI机制原理和使用场景
  15. kibana对logstash监控获取不到数据
  16. Redis的五种数据结构的内部编码
  17. 【LeetCode】 Valid Sudoku
  18. JavaWeb温习之防止表单重复提交
  19. 【idea】常用快捷键
  20. 【HTML5校企公益课】第一天

热门文章

  1. 安卓数据库sqllite查看工具Android Debug Database使用教程
  2. ISCSI共享存储
  3. HGOI 20191106 题解
  4. 「BZOJ 4565」「HAOI 2016」字符合并「区间状压DP」
  5. dup2函数
  6. Kernel Knights (Gym - 101480K)
  7. Linux下更换为阿里yum源
  8. Codeforces 869E. The Untended Antiquity (二维Fenwick,Hash)
  9. 2017 ZSTU寒假排位赛 #5
  10. HTML容器标签和文本标签