原创 s3c2440基于linux的gpio led字符设备驱动实践 全套代码无私奉献

2009-8-24 21:59 6910 7 13 分类: MCU/ 嵌入式

      今天完成了嵌入式linux的第一个驱动的编写和测试,虽然是个简单的程序,但是麻雀虽小,五脏俱全,希望可以给刚开始接触驱动编写的人一些提示,共同进步。


源代码:https://static.assets-stash.eet-china.com/album/old-resources/2009/8/24/7218aa28-452d-43ea-864a-3283206e9ebf.rar


分析如下:


      下面是我的驱动程序:


#include <linux/config.h>//配置头文件
#include <linux/kernel.h>//内核头文件
#include <linux/sched.h>
#include <linux/timer.h>//时钟头文件
#include <linux/init.h>//用户定义模块初始函数名需引用的头文件
#include <linux/module.h>//模块加载的头文件
#include <asm/hardware.h>
#include <asm/arch/S3C2440.h> //这个是2440的寄存器头文件,asm/srch只是个链接


//实际根据自己的情况查找,一般是../../linux2.*.*/include/asm/arch-s3c2440里 编译器


//自己会查询链接,以前不知道,找了半天


// GPIO_LED DEVICE MAJOR
#define GPIO_LED_MAJOR 97    //定义主设备号



//define LED STATUS  我的板子 LED在GPB0 与GPB1 处 大家根据自己情况改 
#define LED_ON  0 //定义LED灯的状态 开
#define LED_OFF 1 //


 



// ------------------- READ ------------------------  这个前面要加static 否则警告
static ssize_t GPIO_LED_read (struct file * file ,char * buf, size_t count, loff_t * f_ops)
{
 return count;


// ------------------- WRITE -----------------------
static ssize_t GPIO_LED_write (struct file * file ,const char * buf, size_t count, loff_t * f_ops)
{
 return count;


// ------------------- IOCTL -----------------------
static ssize_t GPIO_LED_ioctl (struct inode * inode ,struct file * file, unsigned int cmd, long data)         //这个函数实现了led灯亮灭的接口
{
 switch (cmd)
        {
  case LED_ON : { GPBDAT =0x01; break;}  //根据自己情况修改 一个亮 一个灭
  case LED_OFF: { GPBDAT =0x02; break;}  //交替闪烁
                default :
                        {printk ("lcd control : no cmd run  [ --kernel-- ]\n"); return (-EINVAL);}
        }
 return 0;
 
}


// ------------------- OPEN ------------------------
static ssize_t GPIO_LED_open (struct inode * inode ,struct file * file)
{
 MOD_INC_USE_COUNT;
 
 return 0;


// ------------------- RELEASE/CLOSE ---------------
static ssize_t GPIO_LED_release (struct inode  * inode ,struct file * file)
{
 MOD_DEC_USE_COUNT;


 return 0;
}
// -------------------------------------------------



struct file_operations GPIO_LED_ctl_ops ={


 open:  GPIO_LED_open,            //这段赋值代码必须放在接口函数申明之后
 read:  GPIO_LED_read,             //否则编译不过去
 write:  GPIO_LED_write,
 ioctl:  GPIO_LED_ioctl,
 release: GPIO_LED_release,
};


 


// ------------------- INIT ------------------------
static int GPIO_LED_CTL_init(void)
{
    int ret = -ENODEV;
 


 printk("--------------------------------------------\n\n");
 
    GPBCON = 0x0005;   // 设置端口为I/O输出模式
    GPBUP  = 0xff;     // 关闭上拉功能
    GPBDAT = 0xf;      //初始值为高电平熄灭LED灯


    ret = register_chrdev(GPIO_LED_MAJOR, "gpio_led_ctl", &GPIO_LED_ctl_ops);


   //这个驱动注册函数必须放在复制接口的那个结构体之后


 if( ret < 0 )
 {
  printk (" S3C2410: init_module failed with %d\n", ret); 
  return ret;
 }
 else
 {
  printk("S3C2410 gpio_led_driver register success!!! \n");
 }



 return ret;
}



static int __init S3C2410_GPIO_LED_CTL_init(void)
{
    int  ret = -ENODEV;
      
    ret = GPIO_LED_CTL_init();
    if (ret)
      return ret;
    return 0;
}


static void __exit cleanup_GPIO_LED_ctl(void)
{
 unregister_chrdev (GPIO_LED_MAJOR, "gpio_led_ctl" );
 
}
module_init(S3C2410_GPIO_LED_CTL_init);
module_exit(cleanup_GPIO_LED_ctl);


完了编译这个驱动函数


makefile如下:


#################################################
# config


# where the kernel sources are located  这是我的内核头文件的路径 根据自己的修改


KERNEL_DIR := ../../../linux-2.4.20 


 


#################################################
# some magic for using linux kernel settings
# when compiling module(s)


# for new-style kernel Makefiles (2.4)
export-objs := led.o               //要编译好的对象
list-multi :=


obj-m  := led.o


 


here:
 (cd $(KERNEL_DIR); make SUBDIRS=$(PWD) modules) //make
 


clean:
 -rm -f *.o .*.o.flags *~



include $(KERNEL_DIR)/Rules.make               //make的规则 根据自己的情况修改


编译好以后,接下来就是测试是否可以使用驱动了


测试函数如下:


#include <stdio.h>
#include <string.h>
#include <stdlib.h>


#include <fcntl.h>      // open() close()
#include <unistd.h>     // read() write()


#define DEVICE_NAME "/dev/gpio_led_ctl"     //这是设备驱动名字,一会要建立


//define LED STATUS
#define LED_ON  0
#define LED_OFF 1



//------------------------------------- main ---------------------------------------------
int main(void)
{
        int fd;
 int ret;
 char *i;


        printf("\nstart gpio_led_driver test\n\n");


        fd = open(DEVICE_NAME, O_RDWR);
 
 printf("fd = %d\n",fd);
 


        if (fd == -1)
        {
                printf("open device %s error\n",DEVICE_NAME);
        }
        else
        {
  while(1)
  { ioctl(fd,LED_OFF);                 //GPB0亮 GPB1灭
   sleep(1);                                    //等待1秒再做下一步操作
   ioctl(fd,LED_ON);                    //反过来
   sleep(1);


  }
         // close
  ret = close(fd);
  printf ("ret=%d\n",ret);
  printf ("close gpio_led_driver test\n");
        }


        return 0;
}// end main


makefile如下:


CROSS = /opt/host/armv4l/bin/armv4l-unknown-linux-   


 //交叉编译工具路径 根据自己修改
CC = $(CROSS)gcc
AR = $(CROSS)ar
STRIP = $(CROSS)strip



EXEC = test          //生成的可执行文件
OBJS = test.o


all: $(EXEC)


$(EXEC): $(OBJS)
 $(CC) $(LDFLAGS) -o $@ $(OBJS) $(LIBM) $(LDLIBS) $(LIBGCC) -lm  //编译



clean:
 -rm -f $(EXEC) *.elf *.gdb *.o


接下来就是最后的调试了:


首先把生成的led.o和test载到板子上


然后:


insmod led.o  //成功的话,会打印sucess


lsmod  //查看内核里面是否已经有led驱动模块


mknod  /dev/gpio_led_ctl c 97 1  //新建LED的测试设备节点,给test.c使用


//  /dev/gpio_led_ctl 是打开的设备名称,要和测试代码匹配 


// c代表字符设备


// 97是主设备好,与驱动程序匹配  1是从设备号 只有一个选1


最后执行:


./test    //成功了会打印一些 信息 这是你会看到你的板子上 LED交替亮灭 间隔1s

PARTNER CONTENT

文章评论6条评论)

登录后参与讨论

用户221556 2011-10-20 18:20

楼主好人啊

用户380093 2011-9-28 20:09

#include asm/arch/S3C2440.h 在哪里,怎么生成的

用户380093 2011-9-28 20:08

#include #include //这个是2440的寄存器头文件,asm/srch只是个链接 //实际根据自己的情况查找,一般是../../linux2.*.*/include/asm/arch-s3c2440里 编译器 //自己会查询链接,以前不知道,找了半天 在哪里,怎么生成的

用户380093 2011-9-28 20:07

#include #include 在哪里怎么生成的

用户380093 2011-9-28 20:06

#include #include 在哪里,怎么生成的

用户1670261 2010-11-14 11:44

感谢LZ,一直都想学习如何用2440来控制LED,刚找到这篇文章,下午开始照着实践一下,呵呵。
相关推荐阅读
用户539229 2010-01-23 22:52
vmware下debian5的安装配置以及vmtools的安装使用
一.安装vmware6.5.21.下载vmware6.5.2http://4.scdx3.crsky.com/software/vmware_652.rar2.下载vmware6.5.2汉化包http...
用户539229 2009-12-29 12:57
最新诺基亚Qt4.6的上下位机移植手记,触摸屏支持
一.PC for Winxp下的开发环境的搭建诺基亚收购Qt以后,开发了自己的集成开发环境Qt Creater。这是一个非常强大的工具,上手也很快。登陆:http://qt.nokia.com/dow...
用户539229 2009-12-11 20:14
Omap3evm下android开发第一例hello下位机运行实践
1.       驱动的安装<?xml:namespace prefix = o ns = "urn:schemas-microsoft-com:office:office" /> 1.1...
用户539229 2009-12-09 19:01
android的windows下开发环境的搭建与hello第一例
Android的SDK的官方网址如下官方网址:http://androidappdocs.appspot.com/sdk/index.html<?xml:namespace prefix = o...
用户539229 2009-11-28 13:21
Omap3下Goole的Android操作系统的实现 相当于山寨智能机啦
首先要有一个:OMAP3EVM平台的开发板 我用的是OMAP35x的TI的评估板还要有一个SD卡 TI花了很大的功夫打造了OMAP3智能手机开发平台,给用户提供了SDK包,GDK包,以及双核的Davi...
用户539229 2009-11-08 19:50
OMAP35x下OneNand的分析以及x-loader的介绍
OMAP35x下OneNand的分析以及x-loader的介绍<?xml:namespace prefix = o ns = "urn:schemas-microsoft-com:office:...
EE直播间
更多
我要评论
6
7
关闭 站长推荐上一条 /3 下一条