【从零开始,从内核驱动驱动到用户空间调用】编写第一个linux驱动,通过端口访问I/O寄存器。
目的:
通过I/O端口方式访问RTC的秒寄存器;
由于本人从来没看过linux方面的书籍,也只是会在终端用些常用的命令而已,这次老大叫我学着通过I/O端口方式直接去读写寄存器。于是我在google中搜索,得到了一些答案,比如要先申请内存空间,再用ioremap映射到虚拟空间啊之类的。我学着网上的例子,写好了我的第一份代码。编译时竟然找不到头文件,非常头疼,头文件明明在那儿,怎么就找不到呢?在这里非常感谢,linux内核涉及与实现QQ群里面各位师哥师姐的鼎力相助,尽管说什么我都不懂,他们还是非常耐心。在群里大哥的指引下,我有了思路。要么添加系统调用,要么写个驱动。这里我选择了第二种方法。
方案:
引用群里大哥给我画的图片,感谢!
过程:
1. 编写驱动
#include <linux/ioport.h>
#include <linux/module.h>
#include <linux/init.h>
#include <linux/kernel.h>
#include <linux/cdev.h>
#include <linux/moduleparam.h>
#include <linux/slab.h> /* kmalloc() */
#include <linux/fs.h> /* everything... */
#include <linux/errno.h> /* error codes */
#include <linux/types.h> /* size_t */
#include <linux/proc_fs.h>
#include <linux/fcntl.h> /* O_ACCMODE */
#include <linux/seq_file.h>
#include <linux/cdev.h> #include <asm/uaccess.h> /* copy_*_user */
#include <asm/io.h> #define DEVICE_NAME "rtcport"
#define DEVICE_MAJOR 250 #ifndef BCD_TO_BIN
#define BCD_TO_BIN(val) ((val)=((val)&15) + ((val)»4)*10)
#endif dev_t dev = 0;
static struct resource *rtc_resource;
static struct cdev my_dev;
static int RTC_open(struct inode *inode,struct file *filp)
{
printk("open device\n");
return 0;
} static int RTC_close(struct inode *inode,struct file *filp)
{
printk("close device\n");
return 0;
}
static int RTC_read(struct file *filp, char __user *buf, loff_t *f_pos)
{
outb(0,0x70);
int test=inb(0x71);
printk(KERN_DEBUG "second is %02X\n",test);
return 0;
} static int RTC_write(void)
{
return 0;
} static struct file_operations fops={
.owner=THIS_MODULE,
.open=RTC_open,
.release=RTC_close,
.read=RTC_read,
.write=RTC_write,
};
int RTC_init(void)
{
int ret;
//ret=register_chrdev(DEVICE_MAJOR,DEVICE_NAME,&fops);
ret=alloc_chrdev_region(&dev,0,1,DEVICE_NAME);
if (ret < 0) {
printk("RTC: can't get major %d\n", MAJOR(dev));
return ret;
}
printk("Register device successfully!\n");
release_region(0x70, 0x02);
rtc_resource = request_region(0x70,0x02,DEVICE_NAME);
if(rtc_resource == NULL)
{
printk("Unable to register RTC I/O addresses\n");
return -1;
}
cdev_init(&my_dev,&fops);
my_dev.owner=THIS_MODULE;
my_dev.ops=&fops;
ret=cdev_add(&my_dev,MKDEV(MAJOR(dev),0),1);
if(ret<0)
{
printk("RTC: can't add device");
} return 0;
} void RTC_exit(void)
{
// unregister_chrdev(DEVICE_MAJOR,DEVICE_NAME);
// devfs_remove(DEVICE_NAME);
release_region(0x70,0x02);
cdev_del(&my_dev);
unregister_chrdev_region(dev,1);
printk("Device has been unregistered!\n");
} MODULE_LICENSE("GPL");
MODULE_AUTHOR("HJW");
module_init(RTC_init);
module_exit(RTC_exit);
2. Makefile
obj-m += rtc_port.o
ccflags-y=-I/root/testdxx
all:
make $(ccflags-y) -C /lib/modules/$(shell uname -r)/build M=$(PWD) modules clean:
make -C /lib/modules/$(shell uname -r)/build M=$(PWD) clean
说明:ccflags-y是需要包括的头文件的路径,如果不需要依赖自定义的头文件,可去除。
3.make clean(注:若第一次编译,此步骤可略过)
4.make(ls可以看到在路径下多了一些文件)
5. 将模块Insmod进内核
insmod rtc_port.ko
注释:有同学说,诶怎么一点打印信息都没有,原因是printk本身就不会把信息打印到屏幕上,如果有需要的话,大家可以自己去搜索搜索。
6.执行cat /proc/devices可以看到我们新添加的字符型设备rtcport
7.将rtcport创建dev节点
mknod /dev/rtcport c 237 0
8.应用程序app.cpp
#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <sys/ioctl.h>
#include <unistd.h> int main(void)
{
int fd;
char buf[20];
fd=open("/dev/rtcport",O_RDWR);
if (fd<0)
{
perror("open");
return -1;
}
for(int i=0;i<60;i++)
{
read(fd,buf,20);
sleep(1);
}
close(fd);
return 0;
}
9. 输出结果dmesg
10 总结
本文只是介绍了完整的步骤,很多小的知识点都没有提及。下面我将进行概括:
1) 模块的格式
关键函数:Init exit之类的;
2)字符型设备的动态注册(动态注册可以减少设备冲突的概率)
关键函数:
alloc_chrdev_region(&dev,0,1,DEVICE_NAME);
cdev_init;
cdev_add
等等
注:释放的关键函数请参见代码;
3) i/o端口的映射
关键函数:
request_region;
release_region;
小的知识点大家可以边google边对照我的代码,希望这篇文章可以帮助大家少走弯路!
11.展望
下一步是研究IPMI source code,非常渴望能找到志同道合的朋友,大家有过这方面的研究可以给我留言,非常感谢!
【从零开始,从内核驱动驱动到用户空间调用】编写第一个linux驱动,通过端口访问I/O寄存器。的更多相关文章
- linux驱动开发:用户空间操作LCD显示简单的图片【转】
转自:http://blog.csdn.net/changliang7731/article/details/53074616 上一章我们简单介绍了LCD的一些基本原理.当然更深奥的还有,比如gamm ...
- Android系统移植与驱动开发——第六章——使用实例来理解Linux驱动开发及心得
Linux驱动的工作方式就是交互.例如向Linux打印机驱动发送一个打印命令,可以直接使用C语言函数open打开设备文件,在使用C语言函数ioctl向该驱动的设备文件发送打印命令.编写Linux驱动最 ...
- linux驱动开发重点关注内容--摘自《嵌入式Linux驱动模板精讲与项目实践》
本文摘自本人拙著 <嵌入式Linux驱动模板精讲与项目实践> 初步看起来Linux设备驱动开发涉及内容非常多,而须要实现驱动的设备千差万别.事实上做一段时间驱动之后回首看来主要就是下面几点 ...
- 第一个Linux驱动-流水灯【转】
转自:http://www.xuebuyuan.com/1856562.html 水平有限,描述不当之处请指出,转载请注明出处http://blog.csdn.net/vanbreaker/artic ...
- 一个linux 驱动升级的小问题记录
重复踩了两次坑,所以简单记录下. 内核 3.10. 在修改了驱动的gro实现之后,进行驱动版本的升级,make && make install 之后,发现tg3的驱动,没有生效. 相同 ...
- 用Visual Studio 2015 编写第一个UMDF驱动遇到的问题!!
前提:Visual Studio 2015已经成功安装了驱动环境,WDK都已经完全正常安装了,在Visual Studio 2015的菜单可以看到"Driver"菜单项了.这说明已 ...
- Linux 字符设备驱动开发基础(二)—— 编写简单 PWM 设备驱动【转】
本文转载自:https://blog.csdn.net/zqixiao_09/article/details/50858776 版权声明:本文为博主原创文章,未经博主允许不得转载. https: ...
- 用户空间与内核驱动的交互过程 — ioctl
在Linux内核模块的开发过程中,经常涉及到运行在用户空间上的应用程序与内核模块进行交互,ioctl系统调用是常用的一种方式.本文并不涉及vlan的具体原理,仅通过vconfig与vlan内核模块进行 ...
- linux 驱动学习笔记03--Linux 内核的引导
如图所示为 X86 PC 上从上电/复位到运行 Linux 用户空间初始进程的流程.在进入与 Linux相关代码之间,会经历如下阶段. ( 1 ) 当系统上电或复位时, CPU 会将 PC 指针赋值为 ...
随机推荐
- 利用Linux系统函数alarm() 来检测计算机性能
大家都知道,alarm() 是Linux系统自带的定时函数,操作系统管理进程时为每个进程分配了一个定时器,下面利用1秒钟定时,看计算机能计数多少来判断计算机的性能: #include<stdio ...
- 关于Mysql索引的笔记
MySQL索引原理 索引目的 索引的目的在于提高查询效率,可以类比字典,如果要查“mysql”这个单词,我们肯定需要定位到m字母,然后从下往下找到y字母,再找到剩下的sql.如果没有索引,那么你可能需 ...
- 关于java中强制转换
在百度上遇到一个问题,描述如下: 在java中,定义两个变量 byte x = (byte) 128; byte y = (byte)-129; 输出后,为什么结果是-128和128? 借此机会,自己 ...
- Microsoft Jet 数据库引擎找不到对象'Sheet1$_'。请确定对象是否存在,并正确地写出它的名称和路径
We have a CRM add-on for Importing Price Lists into CRM. For this tool, we expect the details to be ...
- 在Windows环境下使用MinGW编译Qt 4.8.6
1.修改环境变量工具推荐:Rapid Environment Editor.官网:http://www.rapidee.com/ 修改前请先备份当前的环境变量.然后: (1)检查系统变量path,删除 ...
- 响应式Web图形篇 —— icon fonts 的探析及应用
前言 像素完美(Pixel Perfection).分辨率无关(Resolution Independent)和多平台体验一致性是设计师们的追求. 可访问性(Accessability).加载性能和重 ...
- 浅谈长尾理论--《Makers》读后感
近期有幸读了一本好书<Makers>,作者是克里斯·安德森.作为3D Robotics和DIY Drones的联合创始人,自然对于正步入的“第三次工业革命”有较为深刻的体会.清晰的逻辑中, ...
- IIS express 7.5 配置和多网站执行
iis express7.5 支持xp 以上的操作系统,能够解决xp.iis的问题. 首先先下载安装iisexpress7.5地址是id=1038">点击打开链接下载完毕点击安装就可以 ...
- nginx学习六 高级数据结构之双向链表ngx_queue_t
1 ngx_queue_t简单介绍 ngx_queue_t是nginx提供的一个轻量级的双向链表容器,它不负责存储数据,既不提供数据的内存分配.它仅仅有两个指针负责把数据链入链表.它跟stl提供的qu ...
- hdu4597 Play Game(DFS)
转载请注明出处:http://blog.csdn.net/u012860063 题目链接:http://acm.hdu.edu.cn/showproblem.php? pid=4597 题意 Alic ...