arm裸机下读写寄存器很容易,各个寄存器和内存的地址是单一地址空间,他们是用相同的指令进行读写操作的.而在linux下就要复杂很多,因为linux支持多个体系架构的CPU。比如arm和x86就不一样,具体的差别我暂时也说不上来,这个涉及到CPU体系的设计。目前我只关心:linux为了支持多个硬件体系,在IO访问上做了自己的接口。可以通过IO内存和IO端口这两种方式进行IO访问。在LED的例子上给出这两种方式的具体实现:

1.利用IO Port的方式:

#include <linux/module.h>
#include <linux/moduleparam.h>
#include <linux/init.h>

#include <linux/kernel.h> /* printk() */
#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 <linux/ioport.h>

#include <mach/regs-gpio.h>
#include <asm/system.h>  /* cli(), *_flags */
#include <asm/uaccess.h> /* copy_*_user */
#include <asm/io.h>

#define LED_NUM   4

struct led_dev
{
 struct cdev dev;
 unsigned port;
 unsigned long offset;
};

struct led_dev led[4];
dev_t dev = 0;
static struct resource *led_resource;

int led_open(struct inode *inode, struct file *filp)
{
 struct led_dev *led; /* device information */

led = container_of(inode->i_cdev, struct led_dev, dev);
 filp->private_data = led; /* for other methods */

return 0;          /* success */
}

int led_release(struct inode *inode, struct file *filp)
{
 return 0;
}

ssize_t led_read(struct file *filp, char __user *buf, size_t count, loff_t *f_pos)
{
 return 0;
}

ssize_t led_write(struct file *filp, const char __user *buf, size_t count, loff_t *f_pos)
{
 char data;
 struct led_dev *led;
 u32 value;
 printk(KERN_INFO "debug by baikal: led dev write\n");
 
 led = (struct led_dev *)filp->private_data;
 copy_from_user(&data,buf,count);
 if(data == '0')
 { 
  printk(KERN_INFO "debug by baikal: led off\n"); 
  value = inl((unsigned)(S3C2410_GPBDAT));
  outl(value | 1<<led->offset,(unsigned)(S3C2410_GPBDAT)); 
  //value = ioread32(led->base);
  //iowrite32( value | 1<<led->offset, led->base);  
 }
 else
 {
  printk(KERN_INFO "debug by baikal: led on\n");
  value = inl((unsigned)(S3C2410_GPBDAT));
  outl(value & ~(1<<led->offset),(unsigned)(S3C2410_GPBDAT)); 
  //value = ioread32(led->base);
  //iowrite32( value & ~(1<<led->offset), led->base);
 }
}

struct file_operations led_fops = {
 .owner =    THIS_MODULE,
 .read =    led_read,
 .write =    led_write,
 //.ioctl =    led_ioctl,
 .open =    led_open,
 .release =  led_release,
};

static int led_init(void)

 int result, i;

result = alloc_chrdev_region(&dev, 0, LED_NUM,"LED");
 if (result < 0) {
  printk(KERN_WARNING "LED: can't get major %d\n", MAJOR(dev));
  return result;
 }
 led_resource = request_region(0x56000014,0x4,"led");
 if(led_resource == NULL)
 {
  printk(KERN_ERR " Unable to register LED I/O addresses\n");
  return -1;
 }
 for(i = 0; i < LED_NUM; i++)
 {
  cdev_init( &led[i].dev, &led_fops);
  //led[i].port = ioport_map(0x56000014,0x4);
  //led[i].base = ioremap(0x56000014,0x4);
  led[i].offset = i + 5;  //leds  GPB5\6\7\8
  led[i].dev.owner = THIS_MODULE;
  led[i].dev.ops = &led_fops;
  result = cdev_add(&led[i].dev,MKDEV(MAJOR(dev),i),1);
  if(result < 0)
  {
   printk(KERN_ERR "LED: can't add led%d\n",i);
   return result;
  }
 }

return 0;
}

static void led_exit(void)
{
 int i;
 release_region(0x56000014,0x4);
 for( i = 0; i < LED_NUM; i++)
 {
  //iounmap(led[i].base);

cdev_del(&led[i].dev); 
 }
 unregister_chrdev_region(dev, LED_NUM);

}

module_init(led_init);
module_exit(led_exit);

MODULE_AUTHOR("Baikal");
MODULE_LICENSE("GPL");
MODULE_DESCRIPTION("Simple LED Driver");

2.利用IO Mem的方式:

#include <linux/module.h>
#include <linux/moduleparam.h>
#include <linux/init.h>

#include <linux/kernel.h> /* printk() */
#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 <linux/ioport.h>

#include <asm/system.h>  /* cli(), *_flags */
#include <asm/uaccess.h> /* copy_*_user */
#include <asm/io.h>

#define LED_NUM   4

struct led_dev
{
 struct cdev dev;
 void __iomem *base;
 unsigned long offset;
};

struct led_dev led[4];
dev_t dev = 0;

int led_open(struct inode *inode, struct file *filp)
{
 struct led_dev *led; /* device information */

led = container_of(inode->i_cdev, struct led_dev, dev);
 filp->private_data = led; /* for other methods */

return 0;          /* success */
}

int led_release(struct inode *inode, struct file *filp)
{
 return 0;
}

ssize_t led_read(struct file *filp, char __user *buf, size_t count, loff_t *f_pos)
{
 return 0;
}

ssize_t led_write(struct file *filp, const char __user *buf, size_t count, loff_t *f_pos)
{
 char data;
 struct led_dev *led;
 u32 value;
 printk(KERN_INFO "debug by baikal: led dev write\n");
 
 led = (struct led_dev *)filp->private_data;
 copy_from_user(&data,buf,count);
 if(data == '0')
 { 
  printk(KERN_INFO "debug by baikal: led off\n");  
  value = ioread32(led->base);
  iowrite32( value | 1<<led->offset, led->base);  
 }
 else
 {
  printk(KERN_INFO "debug by baikal: led on\n");
  value = ioread32(led->base);
  iowrite32( value & ~(1<<led->offset), led->base);
 }
}

struct file_operations led_fops = {
 .owner =    THIS_MODULE,
 .read =    led_read,
 .write =    led_write,
 //.ioctl =    led_ioctl,
 .open =    led_open,
 .release =  led_release,
};

static int led_init(void)

 int result, i;

result = alloc_chrdev_region(&dev, 0, LED_NUM,"LED");
 if (result < 0) {
  printk(KERN_WARNING "LED: can't get major %d\n", MAJOR(dev));
  return result;
 }
 
 for(i = 0; i < LED_NUM; i++)
 {
  cdev_init( &led[i].dev, &led_fops);
  request_mem_region(0x56000014,0x4,"led");
  led[i].base = ioremap(0x56000014,0x4);
  led[i].offset = i + 5;  //leds  GPB5\6\7\8
  led[i].dev.owner = THIS_MODULE;
  led[i].dev.ops = &led_fops;
  result = cdev_add(&led[i].dev,MKDEV(MAJOR(dev),i),1);
  if(result < 0)
  {
   printk(KERN_ERR "LED: can't add led%d\n",i);
   return result;
  }
 }

return 0;
}

static void led_exit(void)
{
 int i;  
 release_mem_region(0x56000014,0x4);
 for( i = 0; i < LED_NUM; i++)
 {
  iounmap(led[i].base);

cdev_del(&led[i].dev); 
 }
 unregister_chrdev_region(dev, LED_NUM);

}

module_init(led_init);
module_exit(led_exit);

MODULE_AUTHOR("Baikal");
MODULE_LICENSE("GPL");
MODULE_DESCRIPTION("Simple LED Driver");

目前,对于具体体系上的linux在移植过程中如何实现这两种方式的方法还不清楚,现在只是会用。等以后有机会了再慢慢理清楚。

Linux下读写寄存器的更多相关文章

  1. Linux下读写芯片的I2C寄存器

    要想在Linux下读写芯片的I2C寄存器,一般需要在Linux编写一份该芯片的I2C驱动,关于Linux下如何编写I2C驱动,前一篇文章<手把手教你写Linux I2C设备驱动>已经做了初 ...

  2. Linux下读写UART串口的代码

    Linux下读写UART串口的代码,从IBM Developer network上拿来的东西,操作比較的复杂,就直接跳过了,好在代码能用,记录一下- 两个实用的函数- //////////////// ...

  3. Linux 下挂在ntfs 硬盘

    CentOS 7 下想要挂载NTFS的文件系统该怎么办呢? 我们需要一个NTFS-3G工具,并编译它之后在mount就可以了,就这么简单. 首先要进入官网下载NTFS-3G工具 http://www. ...

  4. linux 下使用opengl的glut库显示和旋转BMP图片

    效果图: 这里显示的图和原图有明显的色差,目前猜测是opengl渲染时的颜色表顺序跟BMP文件里的颜色表顺序相反导致. BMP里应该是BGRBGRBRG... ,而opengl渲染时应该是按照RGBR ...

  5. 在Linux下如何用Shell脚本读写XML?现有一个config.xml(转)

    在Linux下如何用Shell脚本读写XML?现有一个config.xml <?xml version="1.0" encoding="UTF-8"?&g ...

  6. linux下操作gpio寄存器的方法

    一. 在驱动中: 1. 用的时候映射端口:ioremap; #define GPIO_OFT(x) ((x) - 0x56000000) #define GPFCON (*(volatile unsi ...

  7. linux下c通过虚拟地址映射读写文件的代码

    在代码过程中中,把开发过程中比较好的一些代码片段记录起来,如下的代码内容是关于 linux下c通过虚拟地址映射读写文件的代码,应该对小伙伴有些好处.#include<stdio.h>#in ...

  8. linux下测试磁盘的读写IO速度-简易方法

    linux下测试磁盘的读写IO速度-简易方法 参考资料:https://blog.csdn.net/zqtsx/article/details/25487185 一:使用hdparm命令 这是一个是用 ...

  9. Linux下按扇区读写块设备

    本文介绍Linux下按扇区读写块设备(示例TF卡),实际应用是在Android系统上,主要方法如下: 1.找到sdcard的挂载点,在android2.1系统下应该为/dev/block/mmcblk ...

随机推荐

  1. 21.MFC进制转换工具

    相关代码:链接:https://pan.baidu.com/s/1pKVVUZL 密码:e3vf #include <stdlib.h> #include <stdio.h> ...

  2. 方便查看线程状况的jsp页面

    此方法来自深入理解java虚拟机一书,用作管理员页面,可以随时用浏览器查看线程堆栈 <%@ page language="java" import="java.ut ...

  3. Unix/Linux环境下多一点不如少一点

    正如很多人所知道的$PATH环境变量里存着一张目录列表,当用户要执行某一程序时,系统就会按照列表中的内容去查找该程序的位置.当程序名前不带点斜线 . / 时$PATH就会起作用. 对于普通用户和roo ...

  4. Codefroces 852 G. Bathroom terminal

    G. Bathroom terminal Smith wakes up at the side of a dirty, disused bathroom, his ankle chained to p ...

  5. 联想M4600 (110主板),安装正版win7 ,进入桌面后鼠标无法使用

    问题:联想M4600 (110主板),安装正版win7 ,进入桌面后鼠标无法使用 原因: 110主板和win7系统问题,具体网上查询 处理: 修改bios 中 “USB Virtual KBS Sup ...

  6. Svn备份与Bandizip压缩批处理程序

    目的:为了定时备份多个svn仓库数据,使用批处理程序进行备份并Bandizip进行压缩保存到指定位置,操作完成后弹出成功提示. 为了完成以上目标,需要了解以下几个方面: 批处理命令 Svn命令 Ban ...

  7. yum配置中driver-class-name: com.mysql.jdbc.Driver报错

    错误: 原因: 解决方法:把方框中的<scope>runtime</scope>删掉

  8. linux下安装配置rabbitMQ

    1.安装Erlang 由于RabbitMQ依赖Erlang, 所以需要先安装Erlang Erlang的安装方式大概有两种: 1.从Erlang Solution安装(推荐) # 添加erlang s ...

  9. Springboot 获取yml、properties参数

    获取properties或yml文件的配置数据(两种方法)(默认的application文件或者自定义的yml和properties) 1.使用@Value()注解 1.1 配置数据 如:在prope ...

  10. 【Henu ACM Round #12 E】Thief in a Shop

    [链接] 我是链接,点我呀:) [题意] n个物品,每个物品都有无限个. 第i个物品的价格是一样都,都是ai 让你从中选出恰好k个物品 问你选出的物品的总价值 有多少种不同的可能. [题解] 可以用f ...