字符设备驱动高级篇5——静态映射表的建立过程,动态映射结构体方式操作寄存器

以下内容源于朱有鹏嵌入式课程的学习与整理,如有侵权请告知删除。

补充内容:字符设备驱动基础5——驱动如何操控硬件_天糊土的博客-CSDN博客

一、静态映射表的建立过程

关于“静态映射表的建立”这部分内容,有以下三个关键:

(1)什么是映射表?

所谓映射表,也就是一些关于具体的物理地址、虚拟地址的宏定义。


(2)如何建立映射表?

1)在/x210kernel/arch/arm/mach-s5pv210/mach-smdkc110.c文件中有如下内容:

关于MACHINE_START这个宏的描述,见博客kernel移植——从三星官方内核开始移植,简单地理解,就是这个宏定义了一个struct machine_desc结构体类型的变量,这个结构体变量描述了某个开发板的相关信息,其中就包括静态映射表相关的信息。

这个结构体变量的成员.map_io是一个函数指针,指向smdkc110_map_io函数。该函数位于/arch/arm/mach-s5pv210/mach-smdkc110.c文件中,主要作用是使用(1)中的映射表来建立linux内核的页表映射关系。

2)smdkc110_map_io函数调用关系如下:

|…smdkc110_map_io函数(位于/arch/arm/mach-s5pv210/mach-smdkc110.c)

|………s5p_init_io 函数(位于/arch/arm/plat-s5p/cpu.c)

|……………iotable_init 函数(位于/arch/arm/mm/mmu.c)

 

3)分析变量s5p_iodesc,它是struct map_desc结构体变量。

struct map_desc {unsigned long virtual;//…………虚拟地址unsigned long pfn;//……………………物理地址unsigned long length;//……………映射长度unsigned int type;
};
/* minimal IO mapping */static struct map_desc s5p_iodesc[] __initdata = {{.virtual	= (unsigned long)S5P_VA_CHIPID,.pfn		= __phys_to_pfn(S5P_PA_CHIPID),.length		= SZ_4K,.type		= MT_DEVICE,}, {.virtual	= (unsigned long)S3C_VA_SYS,.pfn		= __phys_to_pfn(S5P_PA_SYSCON),.length		= SZ_64K,.type		= MT_DEVICE,}, {.virtual	= (unsigned long)S3C_VA_UART,.pfn		= __phys_to_pfn(S3C_PA_UART),.length		= SZ_4K,.type		= MT_DEVICE,}, {.virtual	= (unsigned long)VA_VIC0,.pfn		= __phys_to_pfn(S5P_PA_VIC0),.length		= SZ_16K,.type		= MT_DEVICE,}, {.virtual	= (unsigned long)VA_VIC1,.pfn		= __phys_to_pfn(S5P_PA_VIC1),.length		= SZ_16K,.type		= MT_DEVICE,}, {.virtual	= (unsigned long)S3C_VA_TIMER,.pfn		= __phys_to_pfn(S5P_PA_TIMER),.length		= SZ_16K,.type		= MT_DEVICE,}, {.virtual	= (unsigned long)S5P_VA_GPIO,.pfn		= __phys_to_pfn(S5P_PA_GPIO),.length		= SZ_4K,.type		= MT_DEVICE,},
};

经过分析,真正的静态映射表在arch/arm/plat-s5p/cpu.c中的s5p_iodesc这个变量中,该变量的本质是一个结构体数组,数组中每一个元素就是一个映射。这个映射描述了一段物理地址到虚拟地址之间的映射。

这个结构体数组所记录的几个映射关系被iotable_init函数所使用,该函数负责将这个结构体数组格式的表建立成MMU所能识别的页表映射关系。这样在开机后可以直接使用相对应的虚拟地址来访问对应的物理地址。


(3)开机时建立映射表的流程

kernel启动时,smdkc110_map_io如何被调用的?

调用关系如下:

|…………第二阶段的start_kernel函数(位于x210_kernel/init/main.c文件中)

|………………setup_arch函数(位于x210_kernel/arch/arm/kernel/setup.c文件)

|……………………paging_init函数(位于x210_kernel/arch/arm/mm/mmu.c文件中)

|…………………………devicemaps_init函数(该函数部分内容如下)

/* 位于1000行前后* Ask the machine support to map in the statically mapped devices.*/
if (mdesc->map_io) //这里的mdesc是struct machine_desc结构体变量mdesc->map_io();

二、动态映射结构体方式操作寄存器

下面是之前讲解的独立的多次映射:

// 使用动态映射的方式来操作寄存器/*
**request_mem_region参数含义
**第一个参数:起始地址(或者说寄存器的原始地址)
**第二个参数:寄存器的长度(一般32bit,因此4字节)
**第三个参数:这段空间的名字*/
*/                                      if (!request_mem_region(GPJ0CON_PA, 4, "GPJ0CON"))return -EINVAL;if (!request_mem_region(GPJ0DAT_PA, 4, "GPJ0CON"))return -EINVAL;pGPJ0CON = ioremap(GPJ0CON_PA, 4);pGPJ0DAT = ioremap(GPJ0DAT_PA, 4);*pGPJ0CON = 0x11111111;*pGPJ0DAT = ((0<<3) | (0<<4) | (0<<5));		// 亮

下面是真实驱动中,用结构体封装的方法来进行多个寄存器的地址映射。

typedef struct GPJ0REG
{volatile unsigned int gpj0con;volatile unsigned int gpj0dat;
}gpj0_reg_t;#define GPJ0CON		S5PV210_GPJ0CON
#define GPJ0DAT		S5PV210_GPJ0DAT
#define rGPJ0CON	*((volatile unsigned int *)GPJ0CON)
#define rGPJ0DAT	*((volatile unsigned int *)GPJ0DAT)#define GPJ0_REGBASE	0xe0200240 //这个怎么来的?数据手册?gpj0_reg_t *pGPJ0REG;/****省略部分代码****/static int __init chrdev_init(void)
{	printk(KERN_INFO "chrdev_init helloworld init\n");// 在module_init宏调用的函数中去注册字符设备驱动// major传0进去表示要让内核帮我们自动分配一个合适的空白的没被使用的主设备号// 内核如果成功分配就会返回分配的主设备号;如果分配失败会返回负数mymajor = register_chrdev(0, MYNAME, &test_fops);if (mymajor < 0){printk(KERN_ERR "register_chrdev fail\n");return -EINVAL;}printk(KERN_INFO "register_chrdev success... mymajor = %d.\n", mymajor);/*	// 使用动态映射的方式来操作寄存器if (!request_mem_region(GPJ0CON_PA, 4, "GPJ0CON"))return -EINVAL;if (!request_mem_region(GPJ0DAT_PA, 4, "GPJ0CON"))return -EINVAL;pGPJ0CON = ioremap(GPJ0CON_PA, 4);pGPJ0DAT = ioremap(GPJ0DAT_PA, 4);*pGPJ0CON = 0x11111111;*pGPJ0DAT = ((0<<3) | (0<<4) | (0<<5));		// 亮
*/// 2步完成了映射if (!request_mem_region(GPJ0_REGBASE, sizeof(gpj0_reg_t), "GPJ0REG"))return -EINVAL;pGPJ0REG = ioremap(GPJ0_REGBASE, sizeof(gpj0_reg_t));// 映射之后用指向结构体的指针来进行操作// 指针使用->结构体内元素的方式来操作各个寄存器pGPJ0REG->gpj0con = 0x11111111;pGPJ0REG->gpj0dat = ((0<<3) | (0<<4) | (0<<5));		// 亮return 0;
}// 模块卸载函数
static void __exit chrdev_exit(void)
{printk(KERN_INFO "chrdev_exit helloworld exit\n");//	*pGPJ0DAT = ((1<<3) | (1<<4) | (1<<5));	pGPJ0REG->gpj0dat = ((1<<3) | (1<<4) | (1<<5));	// 解除映射
/*iounmap(pGPJ0CON);iounmap(pGPJ0DAT);release_mem_region(GPJ0CON_PA, 4);release_mem_region(GPJ0DAT_PA, 4);
*/iounmap(pGPJ0REG);release_mem_region(GPJ0_REGBASE, sizeof(gpj0_reg_t));// 在module_exit宏调用的函数中去注销字符设备驱动unregister_chrdev(mymajor, MYNAME);
}module_init(chrdev_init);
module_exit(chrdev_exit);// MODULE_xxx这种宏作用是用来添加模块描述信息
MODULE_LICENSE("GPL");				// 描述模块的许可证
MODULE_AUTHOR("aston");				// 描述模块的作者
MODULE_DESCRIPTION("module test");	// 描述模块的介绍信息
MODULE_ALIAS("alias xxx");			// 描述模块的别名信息

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/461273.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

python 分布图_python数据分布型图表柱形分布图系列带误差线的柱形图

柱形分布图系列柱形分布图系列使用柱形图的方式展示数据的分布规律&#xff1b;可以借助误差线或散点图&#xff1b;带误差线的柱形图就是使用每个类别的均值作为柱形的高度&#xff1b;再根据每个类别的标准差绘制误差线&#xff1b;缺点&#xff1a;无法显示数据的分布情况&a…

[汇编] 002基础知识-CPU和寄存器

CPU是什么 当然这里的内存不仅仅指电脑上的内存&#xff0c;例如&#xff1a;我的金士顿8G内存&#xff0c;七彩虹1G独显&#xff0c;在这里来说&#xff0c;显卡也是有内存的(寄存器) CPU如何控制其它部件的&#xff1f; 问题&#xff1a;CPU是如何和电脑主机中其它芯片有条不…

字符设备驱动高级篇6——内核提供的读写寄存器接口

以下内容源于朱有鹏嵌入式课程的学习与整理&#xff0c;如有侵权请告知删除。 1、访问寄存器的方式 之前对寄存器的操作&#xff0c;都是先定义指向寄存器的指针&#xff0c;然后再解引用来对寄存器进行操作。这是因为ARM体系中&#xff0c;内存和IO是统一编址的。但是其他体系…

java台球游戏设计原理_Java实现简单台球游戏

Java实现简单台球桌问题&#xff0c;供大家参考&#xff0c;具体内容如下需求&#xff1a;使小球可以在桌面上移动&#xff0c;移动到桌面边缘将被弹回&#xff0c;显示小区的移动素材&#xff1a;小球照片桌球照片程序源代码&#xff1a;package 桌球游戏;import java.awt.*;i…

关于java assertion

大部分转载自参考资料&#xff1a;http://www.ibm.com/developerworks/cn/java/l-javaassertion/index.html assertion(断言)在软件开发中是一种常用的调试方式&#xff0c;assertion就是在程序中的一条语句&#xff0c;它对一个boolean表达式进行检查&#xff0c;一个正确程序…

IOC是什么?

2019独角兽企业重金招聘Python工程师标准>>> Inversion of Control&#xff0c;即反转控制&#xff0c;或许说为依赖注入更为合适。IoC就是一种设计模式。 Interface Driven Design接口驱动&#xff0c;接口驱动有很多好处&#xff0c;可以提供不同灵活的子类实现&a…

poj2516Minimum Cost

http://poj.org/problem?id2516 建图的时候 有个地方写错了 卡了半年。。 题意看了N久啊 有N个店主需要K种物品 有M个供应点 每个供应点有K种物品 其实是算K次最小费用 然后叠加 分解开来这题就是求把某种物品从供应点送到店主那里 多个源点-》多个汇点 所以加一个超级源点 和…

myeclipse连接mysql怎么调用_myeclipse连接mysql数据库详细步骤

第一步 打开Database windows-prefenrence-showview-DBbrowser ,此时会在工具底部有个DBbrowser &#xff0c;选中它&#xff0c;再它所控制的页面的任意位置 右击new---跳转到一个配置driver的页面 (选择连接方式)图一打开myeclipse然后点击window窗口 点击Open Perspective…

虚拟内存管理

MMU 现代操作系统普遍采用虚拟内存管理&#xff08;Virtual Memory Management&#xff09;机制&#xff0c;这需要处理器中的MMU&#xff08;Memory Management Unit&#xff0c;内存管理单元&#xff09;提供支持&#xff0c;本节简要介绍MMU的作用。 首先引入两个概念&…

mysql重新用户设置密码_mysql用户密码如何重新设置?

mysql用户密码重新设置停掉MySQL服务&#xff1a;sudo service mysql stop以上命令适用于Ubuntu和Debian。CentOS、Fedora和RHEL下使用mysqld替换mysql。以安全模式启动mysql&#xff1a;sudo mysqld_safe --skip-grant-tables --skip-networking &这样我们就可以直接用roo…

第三章 门电路

1 半导体二极管开关特性 1 二极管的特性可以近似的用3.2.1的PN结方程和图3.2.2伏安特性曲线描述 如下图 二极管近似伏安特性和对应的等效电路 1 a电路表示vcc和r都很小时候二极管正向导通压降和正向电阻都不能忽视 2 b电路表示二极管正向导通电压不可以忽视&#xff0c;但是二…

mysql查询数据库日期_mysql如何查询日期与时间

前言&#xff1a;在项目开发中&#xff0c;一些业务表字段经常使用日期和时间类型&#xff0c;而且后续还会牵涉到这类字段的查询。关于日期及时间的查询等各类需求也很多&#xff0c;本篇文章简单讲讲日期及时间字段的规范化查询方法。1.日期和时间类型概览MySQL支持的日期和时…

设备驱动框架3——使用gpiolib完成LED驱动

以下内容源于朱有鹏嵌入式课程的学习整理&#xff0c;如有侵权请告知删除。 一、前言 在实际情况中&#xff0c;很多硬件都要用到GPIO&#xff0c;因此GPIO会复用&#xff1b;如果同一个GPIO被2个驱动同时控制就会出现bug&#xff1b;因此内核提供了gpiolib来统一管理系统中所有…

from PyQt4 import QtGui,QtCore出错-解

from PyQt4 import QtGui,QtCore出错-解今天尝试着安装PyQt写界面&#xff0c;官网下载后发现import出错了&#xff0c;情况如下图&#xff1a;import PyQt4就可以&#xff0c;from PyQt4 import QtCore却不行提示DLL load faied找了下网上有些人说是某些dll文件丢失了&#xf…

设备驱动框架4——将驱动集成到内核中

以下内容源于朱有鹏嵌入式课程的学习与整理&#xff0c;如有侵权请告知删除。 驱动集成到内核的概念 驱动开发的步骤一般是&#xff1a; &#xff08;1&#xff09;以模块的形式在内核外部编写与调试 &#xff08;2&#xff09;将调试好的驱动代码集成到kernel中 之前我们编写的…

例子简单说说C# ref和out

首写从这字段看 ref 就是引用的意思 out当然就是输出了public void getRefStr(ref string str) {str"hello 你好&#xff0c;你变成了Ref了" }public void getOutStr(out string outStr){outStr "hello 你好&#xff0c;你是out输出的值";} protected…

VARIANT变体类型数据

2019独角兽企业重金招聘Python工程师标准>>> 特殊 Variant 是一种特殊的数据类型&#xff0c;除了定长String数据及用户定义类型外&#xff0c;可以包含任何种类的数据。Variant 也可以包含Empty、Error、Nothing及Null等特殊值。可以用VarType函数或TypeName函数来…

mysql修改校对集_MySQL 图文详细教程之校对集问题

软件安装&#xff1a;装机软件必备包SQL是Structured Query Language(结构化查询语言)的缩写。SQL是专为数据库而建立的操作命令集&#xff0c;是一种功能齐全的数据库语言。在使用它时&#xff0c;只需要发出“做什么”的命令&#xff0c;“怎么做”是不用使用者考虑的。SQL功…

C# winform 魔兽MH全图制作教程(1): 开发准备工作

C# winform 魔兽MH全图制作教程&#xff08;1&#xff09;: 开发准备工作 一、开发条件&#xff1a; Visual Studio 2008win xp,win 7,win 2003.C# 语言基础会调试能够运行游戏&#xff1a;《魔兽争霸3冰封王座》拥有版本魔兽客户端版本切换器1.20E,1.24E,1.24D二、设计思路&am…

从常识看中国经济社会-再续之续:套利

2019独角兽企业重金招聘Python工程师标准>>> 《全球化掠夺》提及财富流转的路径&#xff0c;世界仍旧是个丛林&#xff0c;每个人、每个族群都在争夺自己的利益。在一个经济体的内部&#xff0c;财富是垂直流动的&#xff1b;在全球化的经济体中&#xff0c;财富是纵…