FL2440字符设备驱动之LED

今天开始我们来学习linux驱动的开发,驱动分为字符设备驱动,块设备,网络设备驱动,讲这个之前我说一下我用的虚拟机版本和LInux内核版本,开始我用的redhat 9.0  开始用的好好的,到后来自己编译busybox的时候总是出错误,这个期间我尝试了很多次,在网上也找到了很多方法,可还是解决不了问题,后来找到了原因是虚拟机版本的问题,后来我换了ubuntu9.10,那些错误就没有了。

在这里我我在说明一下FL2440光盘自带的Linux2.6.28.7是不是支持自动创建设备节点的,需要在/etc/init.d/rcS中添加  mdev -s 。学习的同时可以参考韦东山老师的视频,讲得非常好,注意下我们的代码和他的代码有一点区别,就是一些函数可能用的不同(是内核版本的问题),以后我讲的驱动都是用的Linux2.6.28.7,这样也方便你们学习。

这些代码你直接copy上去都是可以用的废话不多说了,直接上代码:

/s3c2440_leds.c
//#include <linux/config.h>
#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/fs.h>
#include <linux/init.h>
#include <linux/device.h>
//#include <linux/devfs_fs_kernel.h>
#include <linux/miscdevice.h>
#include <linux/delay.h>
#include <asm/irq.h>
#include <asm/io.h>
#include <asm/uaccess.h>
#include <mach/regs-gpio.h>
#include <mach/hardware.h>
#include <linux/device.h>
#include <linux/gpio.h>

#define DEVICE_NAME "leds"
#define LED_MAJOR 231
#define LED_OFF         0
#define LED_ON         1
#define ALL_LED_OFF      3
#define ALL_LED_ON       4

MODULE_LICENSE("Dual BSD/GPL");

static struct class *led_2;

static unsigned long led_table [] = {
S3C2410_GPB5,
S3C2410_GPB6,
S3C2410_GPB8,
S3C2410_GPB10,
};
/*等价于设置GPFCON*/
static unsigned int led_cfg_table [] = {
S3C2410_GPB5_OUTP, //0x01<<10 defined in refg-gpio.h
S3C2410_GPB6_OUTP,
S3C2410_GPB8_OUTP,
S3C2410_GPB10_OUTP,
};

static int s3c2440_leds_ioctl( struct inode *inode, struct file *file,unsigned int cmd,unsigned long arg)
{
     int i;
if (arg > 4)
{
return -EINVAL;
}
switch(cmd)
{
case LED_ON:  //set the pin
s3c2410_gpio_setpin(led_table[arg], 0);
break;

case LED_OFF:  //clr the pin
s3c2410_gpio_setpin(led_table[arg], 1);
break;

case ALL_LED_ON:  //set all pin
for (i = 0; i < 4; i++)
s3c2410_gpio_setpin(led_table[i], 0);
break;

case ALL_LED_OFF:  //clr all pin
for (i = 0; i < 4; i++)
s3c2410_gpio_setpin(led_table[i], 1);
break;

default:
return -EINVAL;
}
}

static struct file_operations s3c2440_leds_fops = {
.owner = THIS_MODULE,
.ioctl = s3c2440_leds_ioctl,
};

 static int major;
static int __init s3c2440_leds_init(void)
{

  int i;
major = register_chrdev(0, DEVICE_NAME, &s3c2440_leds_fops);
if (major < 0) 
{
   printk(DEVICE_NAME " can't register major number\n");
   return  major;
}
       led_2 = class_create(THIS_MODULE, DEVICE_NAME);
device_create(led_2, NULL, MKDEV(major, 0), NULL, "led_2"); 

for (i = 0; i < 4 ; i++)
{
s3c2410_gpio_cfgpin(led_table[i], led_cfg_table[i]);
s3c2410_gpio_setpin(led_table[i], 1); 
}
printk(DEVICE_NAME " initialized\n");
return 0;
}
static void __exit s3c2440_leds_exit(void)
{
unregister_chrdev(major, DEVICE_NAME);
device_destroy(led_2,MKDEV(major, 0));
class_destroy(led_2);
}
module_init(s3c2440_leds_init);
module_exit(s3c2440_leds_exit);

测试程序:

#include "stdio.h"
#include "unistd.h"
#include "sys/types.h"
#include "sys/ioctl.h"
#include "stdlib.h"
#include "termios.h"
#include "sys/stat.h"
#include "fcntl.h"
#include "sys/time.h"

int main(int argc, char **argv)
{
unsigned int on;
unsigned int led_num;
int fd;
fd=open("/dev/led_2",O_RDWR);//打开文件返回值是个非负值
if (fd < 0)
{
printf("open device led");
      return -1;
}

      printf("%s <on/off>\n",argv[0]);
  
if (argc == 2)

{
sscanf(argv[1], "%d", &on);
if (on < 2)
{
ioctl(fd, on+3, 0);
}
else
{
printf("Usage: led led_num 0|1  or  led 0|1\n");
exit(1);
}
}

if (argc == 3)
{
sscanf(argv[1], "%d", &led_num);
sscanf(argv[2], "%d", &on);
if ((on < 2) && (led_num>0 || led_num < 4))
ioctl(fd, on, (led_num-1));
else
{
printf("Usage: led led_num 0|1  or  led 0|1\n");
exit(1);
}
}

close(fd);
return 0;
}

在编写个Makefile:

obj-m :=test.o
KERNELDIR ?= /home/work/Linux/linux-2.6.28.7//内核路径
PWD := $(shell pwd)
default:
$(MAKE) -C $(KERNELDIR) M=$(PWD) modules
clean:
rm -f  *o  *.mod.o  *mod.c  *.symvers *.order

时间: 2024-11-05 19:03:25

FL2440字符设备驱动之LED的相关文章

FL2440字符设备驱动开发之查询按键

上一篇连接地址:http://blog.csdn.net/qq_21792169/article/details/48414687 驱动程序: #include <linux/module.h> #include <linux/kernel.h> #include <linux/fs.h> #include <linux/init.h> #include <linux/device.h> //#include <linux/devfs_fs

Linux字符设备驱动编写基本流程

  ---简介 Linux下的MISC简单字符设备驱动虽然使用简单,但却不灵活. 只能建立主设备号为10的设备文件.字符设备比较容易理解,同时也能够满足大多数简单的硬件设备,字符设备通过文件系          统中的名字来读取.这些名字就是文件系统中的特殊文件或者称为设备文件.文件系统的简单结点,一般位于/dev/目录下          使用ls进行查看会显示以C开头证明这是字符设备文件crw--w---- 1 root tty 4, 0 4月 14 11:05 tty0.第一个数字是主设备

【Linux驱动】字符设备驱动

一.linux系统将设备分为3类:字符设备.块设备.网络设备.使用驱动程序: 1.字符设备:是指只能一个字节一个字节读写的设备,不能随机读取设备内存中的某一数据,读取数据需要按照先后数据.字符设备是面向流的设备,常见的字符设备有鼠标.键盘.串口.控制台和LED设备等. 2.块设备:是指可以从设备的任意位置读取一定长度数据的设备.块设备包括硬盘.磁盘.U盘和SD卡等. 每一个字符设备或块设备都在/dev目录下对应一个设备文件.linux用户程序通过设备文件(或称设备节点)来使用驱动程序操作字符设备

《嵌入式Linux开发实用教程》——4.2 字符设备驱动

4.2 字符设备驱动 Linux操作系统将所有的设备都会看成是文件,因此当我们需要访问设备时,都是通过操作文件的方式进行访问.对字符设备的读写是以字节为单位进行的. 对字符设备驱动程序的学习过程,主要以两个具有代表性且在OK6410开发平台可实践性的字符驱动展开分析,分别为LED驱动程序.ADC驱动程序. 4.2.1 LED驱动程序设计 为了展现LED的裸板程序和基于Linux系统的LED驱动程序的区别与减少难度梯度,在写LED驱动程序之前很有必要先看一下LED的裸板程序是怎样设计的. 1.LE

Linux字符设备驱动之cdev_init()

1.内核中每个字符设备都对应一个 cdev 结构的变量,下面是它的定义: linux-2.6.22/include/linux/cdev.h struct cdev {  13        struct kobject kobj;  14        struct module *owner;  15        const struct file_operations *ops;  16        struct list_head list;  17        dev_t dev

应用程序-再次请求支援,关于字符设备驱动的程序问题,求帮找错(不好意思没点数了)

问题描述 再次请求支援,关于字符设备驱动的程序问题,求帮找错(不好意思没点数了) 先贴代码,问题在后面 驱动函数pressure.c: /* 功能: 外部中断4接一个人体红外感应,如果感应到有人体活动(红外模块输出高电平),如果感应不到 人体活动(红外模块输出低电平)(变量red) 外部中断8接了一个按钮开关,如果按钮开关被按下,则输出为低电平,外部中断8管脚应该低电平, 反之,高电平 (变量pre) 只有当设备感应到有人体活动同时检测到按钮开关被按下.即 red == 1,pre == 0,的

Linux字符设备驱动框架

字符设备是Linux三大设备之一(另外两种是块设备,网络设备),字符设备就是字节流形式通讯的I/O设备,绝大部分设备都是字符设备,常见的字符设备包括鼠标.键盘.显示器.串口等等,当我们执行ls -l /dev的时候,就能看到大量的设备文件,c就是字符设备,b就是块设备,网络设备没有对应的设备文件.编写一个外部模块的字符设备驱动,除了要实现编写一个模块所需要的代码之外,还需要编写作为一个字符设备的代码. 驱动模型 Linux一切皆文件,那么作为一个设备文件,它的操作方法接口封装在struct fi

Linux字符设备驱动之异步通知

学习到这里到这里我们觉得这下这个驱动应该就很完善了吧,但是不仅然,我们是不是想当有按键按下的时候,这个时候再去通知用户空间的read函数来读,这样是不是更方便的都,免得函数也老是在哪里休眠.在这里说下:我是不会讲代码的,这些代码比较简单,这只是一些基础的字符设备驱动驱动,到以后我也会讲一些高级点的驱动.这些代码已经讲得很清楚了,全是本人自己写的,是没有错误的,只是自己写的时候注意下自己的内核版本就可以了,以免有一些函数是不能被发现. 上一篇文章的链接:http://blog.csdn.net/q

Linux字符设备驱动

1.预备知识: 应用程序.库.内核.驱动程序的关系 应用程序调用应用程序函数库完成功能应用程序以文件形式访问各种资源应用程序函数库部分函数直接完成功能     部分函数通过系统调用由内核完成   内核处理系统调用,调用设备驱动程序   设备驱动直接与硬件通信 设备类型 字符设备     对字符设备发出读/写请求时,实际的硬件I/O操作一般紧接着发生 块设备     块设备与之相反,它利用系统内存作为缓冲区 网络设备     网络设备是一类特殊的设备,它不像字符设备或块设备那样通过对应的设备文件节