原创 mini2440的led驱动

2009-8-31 20:18 5517 10 12 分类: 软件与OS

友善之臂mini2440<?xml:namespace prefix = o ns = "urn:schemas-microsoft-com:office:office" />


linux-<?xml:namespace prefix = st1 ns = "urn:schemas-microsoft-com:office:smarttags" />2.6.29.1


arm-linux-gcc-4.3.2


 


自己写的mini2440led驱动和测试程序,下面是添加注释的源码和makefile。


//s3c24xx_leds.c


#include <linux/module.h>  //模块加载的头文件


#include <linux/kernel.h>   // 内核头文件


//所以模块代码都包含以上两个头文件


#include <linux/fs.h>      //定义file_operations结构


#include <linux/init.h>   //用户定义的模块初始化函数需要引用的头文件


#include <linux/delay.h>  //延时函数的头文件


#include <mach/regs-gpio.h>  //定义GPIO寄存器相关的宏


#include <mach/hardware.h>  //硬件相关的函数


 


#define DEVICE_NAME     "leds" 


#define LED_MAJOR       231    


 


#define IOCTL_LED_ON    0


#define IOCTL_LED_OFF   1


//连接ledGPIO管脚的宏,定义在/arch/arm/mach-s3c2410/include/mach/regs-gpio.h


static unsigned long led_table [] = {      


    S3C2410_GPB5,


    S3C2410_GPB6,


    S3C2410_GPB7,


    S3C2410_GPB8,


};


//设置GPIO管脚为输出的宏,定义在/arch/arm/mach-s3c2410/include/mach/regs-gpio.h


static unsigned int led_cfg_table [] = {


    S3C2410_GPB5_OUTP,


    S3C2410_GPB6_OUTP,


    S3C2410_GPB7_OUTP,


    S3C2410_GPB8_OUTP,


};


//驱动接口函数open的用户实现。


static int s3c24xx_leds_open(struct inode *inode, struct file *file)


{


    int i;


   


for (i = 0; i < 4; i++) {


//设置gpio管脚状态的函数,定义在mach/hardware.h,参数的宏定义在mach/hardware.h


        s3c2410_gpio_cfgpin(led_table, led_cfg_table);


    }


    return 0;


}


//驱动接口函数ioctl的用户实现。


static int s3c24xx_leds_ioctl(


    struct inode *inode,


    struct file *file,


    unsigned int cmd,


    unsigned long arg)


{


    if (arg > 4) {


        return -EINVAL;


    }


   


    switch(cmd) {


case IOCTL_LED_ON:


//设置gpio管脚输出电平的函数,定义在mach/hardware.h


        s3c2410_gpio_setpin(led_table[arg], 0);


        return 0;


 


    case IOCTL_LED_OFF:


        s3c2410_gpio_setpin(led_table[arg], 1);


        return 0;


 


    default:


        return -EINVAL;


    }


}


 


static struct file_operations s3c24xx_leds_fops = {


.owner  =   THIS_MODULE,  


//指向编译模块是自动创建的__this_module变量的宏,无需关注


    .open   =   s3c24xx_leds_open,    


    .ioctl  =   s3c24xx_leds_ioctl,


};


static int __init s3c24xx_leds_init(void)


{


    int ret;


    ret = register_chrdev(LED_MAJOR, DEVICE_NAME, &s3c24xx_leds_fops);


    if (ret < 0) {


      printk(DEVICE_NAME " can't register major number\n");


      return ret;


    }


    printk(DEVICE_NAME " initialized\n");


    return 0;


}


static void __exit s3c24xx_leds_exit(void)


{


    unregister_chrdev(LED_MAJOR, DEVICE_NAME);


}


module_init(s3c24xx_leds_init);


module_exit(s3c24xx_leds_exit);


//end


 


应用程序执行的openinctl等系统调用,它们的参数与驱动程序中相应函数的参数不是一一对应的,其中经过了内核文件系统层的转换。


系统调用函数的原型如下:


int open(const char *pathname, int flags);


int ioctl (int d, int request, …);


ssize_t read(int fd, void *buf, size_t count);


ssize_t write(int fd, const void *buf, size_t count);


file_operations结构中的成员如下:


int (*open) (struct inode *inode, struct file *filp);


int (*ioctl) (struct inode *inode, struct file *filp, unsigned int cmd, unsigned long arg);


ssize_t (*read) (struct file *filp, char __user *buff, size_t count, loff_t *offp);


ssize_t (*write) (struct file *filp, const char __user *buff, size_t count), loft *offp);



1.       系统调用的open函数只需传入设备文件路径和打开状态;


2.       系统调用的ioctl函数参数个数可变,一般为三个,后面两个参数与file_operations结构中ioctl成员的后两个参数对应;


3.       系统调用的read函数传入的bufcount,对应file_operations结构中的bufcount参数,而offp表示用户在文件中进行存取操作的位置,当执行玩读写操作后有驱动程序设置。


4.       系统调用的write函数与read类似。


 


Makefile


CROSS=arm-linux-


KERNELDIR = /home/bob/linux_src/linux-2.6.29.1


#定义内核源码的路径


PWD := $(shell pwd)


#获取当前路径


.PHONY: modules clean


#指明modulesclean为伪目标


obj-m += s3c24xx_leds.o


#表示要构造的模块名为s3c24xx_leds.ko


modules:


       $(MAKE) -C $(KERNELDIR) M=$(PWD) modules


#-C:进入$(KERNELDIR)后再执行make,读取那里的Makefile


#M=$(PWM):然后返回当前的位置(驱动源代码的位置)继续执行当前的Makefile


clean:


       rm -rf *.o *~ core .depend .*.cmd  *.mod.c .tmp_versions



#end


测试代码:


                  

https://static.assets-stash.eet-china.com/album/old-resources/2009/8/31/c932d790-4339-4e98-bc24-05d3c6827cfb.rar

文章评论2条评论)

登录后参与讨论

用户539229 2009-9-6 10:18

上面的可以的啊 你在2410的头文件里面加上定义寄存器的地址: 方法和裸机差不多 比如: #define GPBCON *voilate(...#地址) (具体咋写忘了) 然后就可以直接写寄存器了 switch (cmd) { case LED_ON : { GPBDAT =0x01; break;} } return 0; 可以参考的程序 http://blog.ednchina.com/gurongjiang/254851/message.aspx

huotingtu_505472073 2009-9-1 10:37

要是楼主能不用2410的IO操作函数而直接用Linux系统的IO操作就好了,我现在就是郁闷在直接IO操作上不晓得怎么弄了。谢谢楼主共享资料和经验
相关推荐阅读
用户190932 2011-02-27 14:34
剖析Linux系统启动过程
本文以RedHat9.0和i386平台为例,剖析了从用户打开电源直到屏幕出现命令行提示符的整个Linux启动过程。并且介绍了启动中涉及到的各种文件。   阅读Linux源代码,无疑是深入学习Linu...
用户190932 2011-02-26 23:18
RedHat系列开机启动脚本顺序
RedHat系列Linux开机启动脚本顺序如果服务器重启之后还需要手工开启相关服务、那还说明上次的配置工作未完成,现特地总结了下red hat linux下开机自动启动脚本所涉及的知识和方法、如下:一...
用户190932 2011-02-05 13:10
2011第一篇:一个程序员的奋斗历程
转帖:这是多年前一位程序员所写,共勉! 我上的是一个三流的高校,就连同一个城市的人多数都不知道。因为学校不好也就没有指望能靠学校名气找一个好工作。所有的希望都寄托在自己的努力上了,大一开学前的假期我...
用户190932 2010-10-06 23:41
man手册的用法
使用Linux下man已经很久了。但只是大概知道其内容的存放目录,以及man number xxx中mumber的含义。今天需要添加C++ man page。顺便将man的内容作一个较详细的了解。0....
用户190932 2010-04-28 14:05
ubuntu下mini2440的NFS挂载
本文转载自:http://hi.baidu.com/linuxcfan/blog/item/2c4c7e39723194c8d56225b4.html网络文件系统(NFS)一、NFS简介1、NFS就是...
用户190932 2010-04-14 22:56
关于x86、i386、i486、i586和i686等名词的解释
一、x86与i386、i486、i586、i686等        x86或80x86是英特尔Intel首先开发制造的一种微处理器体系结构的泛称。 该系列较早期的处理器名称是以数字来表示,并以“86...
我要评论
2
10
关闭 站长推荐上一条 /2 下一条