[ARM应用]LED设备驱动实例

发布者:MysticJourney最新更新时间:2016-11-22 来源: eefocus关键字:ARM  LED  设备驱动 手机看文章 扫描二维码
随时随地手机看文章

    通过一个最简单的LED驱动实例来展示字符设备驱动程序设计,开发及测试的整个过程。

1. 实验目的

(1)了解LED原理及其与S3C2410的接口电路设计。

(2)了解S3C2410芯片的I/O端口配置方法。

(3)通过S3C2410芯片的GPF4端口控制板上D1的亮灭。

(4)掌握LED驱动的编写及测试过程,进而熟悉开发简单字符设备驱动的方法。

 

2. 实验原理

2.1 S3C2410的GPIO概述

打开S3C2410的数据手册,看看它的I/O端口,共有117只多功能输入/输出端口,它们分别为:

GPA口:23只输出口

GPB口:11只输入/输出口

GPC口:16只输入/输出口

GPD口:16只输入/输出口

GPE口:16只输入/输出口

GPF口: 8只输入/输出口

GPG口:16只输入/输出口

GPH口:11只输入/输出口

每组端口都有复用的功能,例如作为输入口或输出口,还可以定义为中断触发功能,我们可以通过配置这些端口的控制寄存器来满足不同系统和设计的需要,下面马上就会说到如何配置端口控制寄存器,数据寄存器等。在运行主程序之前,必须先对每一个用到的引脚的功能进行设置。如果某些引脚的复用功能没有使用,那么可以先将该引脚设置为I/O口。

2.2 GPIO对应的特殊功能寄存器

?         端口控制寄存器(GPFCON)

在S3C2410中,大多数的引脚都是复用的。因此,需要通过控制寄存器来配置每个I/O引脚的功能。如果GPF0~GPF7和GPG0~GPG7用做断电模式下的唤醒信号,这些端口必须配置成中断模式。

下表是GPFCON寄存器的配置说明:GPFCON中每两位对应一个端口,每个端口可以配置三种功能。如果某个端口所对应的两位配成00,则该端口作为输入,01为输出,10为外部中断功能,11保留。

GPFCON

BIT

描述

GPF7

[15:14]

00:Input, 01:Output, 10:EINT7,11:Reserved

GPF6

[13:12]

00:Input, 01:Output, 10:EINT6,11:Reserved

GPF5

[11:10]

00:Input, 01:Output, 10:EINT5,11:Reserved

GPF4

[9:8]

00:Input, 01:Output, 10:EINT4,11:Reserved

GPF3

[7:6]

00:Input, 01:Output, 10:EINT3,11:Reserved

GPF2

[5:4]

00:Input, 01:Output, 10:EINT2,11:Reserved

GPF1

[3:2]

00:Input, 01:Output, 10:EINT1,11:Reserved

GPF0

[1:0]

00:Input, 01:Output, 10:EINT0,11:Reserved

表2.1  GPF端口控制寄存器

GPA,GPB等端口都有相应的控制寄存器,它们的配置说明也类似GPF。你在开发过程中用到这些端口的时候,可以参考s3c2410的数据手册。

?         端口数据寄存器(GPFDAT)

如果端口被配置成输出口,那么输出数据可以写入GPnDAT相应的位;如果端口定义为输入口,那么输入数据可以从GPnDAT相应的位读入。如果端口定义为功能引脚(如外部中断),从该寄存器中读出来的值是未定义的。

GPFDAT

BIT

描述

GPFDAT [7:0]

[7:0]

When the port is configured as input port,data from externel sources can be read to the corresponding pin.When the port is configured as output port,data written in this register can  be sent to the corresponding pin.When the port is configured as functional pin,undefined value will be read.

表2.2  GPF端口数据寄存器

?         端口上拉寄存器(GPFUP)

当GPF口作为输入口时,还可以设置内部上拉电阻,其定义如表2.2.3所示。端口上拉寄存器控制每个端口的上拉电阻的使能和关闭。当相应位为“0”时,上拉电阻使能,当相应位为“1”时,上拉电阻关闭。

GPFUP

BIT

描述

GPFUP [7:0]

[7:0]

0:the pull-up function attached to the corresponding port pin is enabled.

1:the pull-up function is disabled.

表2.2.3  GPF端口上拉寄存器

3. 实验任务

本实验的最终目的是实现S3C2410平台上的LED驱动。包含以下几个任务:

(1)编写LED设备驱动程序,驱动程序中手动定义设备名称及主设备号,这个设备号必须是系统尚未使用的设备号,这里暂用212,你也可以尝试用别的数字。当然最好的方法是动态分配设备号,如果使用这种方法,在/dev目录下为LED设备建立设备节点的时候,需要先从/proc/devices文件中获取该设备的主设备号。在驱动程序中还要实现与LED相应的I/O配置,以及读写设备的接口函数等。

(2)将驱动编译成模块,并实现模块的加载及卸载。

(3)编写驱动的测试程序,在程序中实现打开LED设备,控制LED设备一亮一灭,关闭LED设备等。

 

4. 实验步骤

以下操作都在nfs文件系统目录(/home/kernel/rootfs/rootfs)下进行,因此先执行如下命令。

cd /home/kernel/rootfs/rootfs

(1)编写led.c文件

建立led目录:

mkdir usr/led

进入led目录,在该目录下建立两个子目录driver 和test ,前者用来存放驱动程序,后者用来存放驱动测试程序:

cd usr/led

mkdir driver test

进入驱动程序目录,建立设备驱动文件led.c:

cd driver 

vi led.c

LED驱动程序如下led.c所示:

//***************************** 头文件 ********************************

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

//*********************** 定义设备结构体及相关宏 ***************************

#define DEVICE_NAME    "led"           //定义设备名

#define DEVICE_MAJOR  212    //手动定义LED设备的主设备号为212

static int led_major = DEVICE_MAJOR ;

 

#define LED1        S3C2410_GPF4   //定义LED1对应S3C2410的GPF4端口

#define LED1_OUTP    S3C2410_GPF4_OUTP

 

#define LED_ON   0  //给端口低电平(0)时,LED亮

#define LED_OFF  1  //给端口高电平(1)时,LED灭

 

//定义LED设备结构体

struct s3c2410_led_dev

{

       struct cdev cdev;  //LED设备对应一个字符设备结构体

       int status;  //LED状态标识,0代表灭,1代表亮

};

static struct s3c2410_led_dev dev;

 

//***************************** 函数声明 ********************************

void s3c2410_led_InitIO(void);  //初始化IO端口的函数

 

//***************************** 函数定义 ********************************

/*

======================================================================

s3c2410_led _InitIO()

 描述   : 初始化IO端口

 参数   : 无

 返回值 : 无

======================================================================

*/

void s3c2410_led_InitIO(void)

{

int i;

    //配置LED对应的端口为输出          

s3c2410_gpio_cfgpin(LED1, LED1_OUTP);

    //配置LED初始为熄灭状态

s3c2410_gpio_setpin(LED1, LED_OFF);

}

 

/*

======================================================================

s3c2410_led_open()

 描述    : 打开设备

 参数      :

 返回值  : 0

======================================================================

*/

static int s3c2410_led_open(struct inode *inode,struct file *filp)

{    

return 0;

}

 

/*

======================================================================

s3c2410_led_release()

 描述    : 注销设备

 参数    : 

 返回值  : 0

======================================================================

*/

static int s3c2410_led_release(struct inode *inode,struct file *filp)

{

    return 0;

}

 

/*

======================================================================

s3c2410_led_ioctl()

 描述   :  IO控制函数,通过LED_ON和LED_OFF命令控制LED的亮灭

 参数   :  cmd:用户控制命令,包括LED_ON和LED_OFF

 返回值 :  0

======================================================================

*/

static int s3c2410_led_ioctl(struct inode *inode,struct file *filp,

                                unsigned int cmd,unsigned long arg)

{

       switch(cmd) {

         case LED_ON:

s3c2410_gpio_setpin(LED1, LED_ON);

              dev.status = 1;

break;

           case LED_OFF:

s3c2410_gpio_setpin(LED1, LED_OFF);

dev.status = 0;

break;

         default:

              return -EINVAL;

}

return 0;

}

 

/*

======================================================================

s3c2410_led_read()

 描述   :  读函数,读取LED的状态

 参数   :  buffer: 用来存储读取的LED状态;

              count: 用来记录用户读取了多少个字符

 返回值 :  count

======================================================================

*/

static ssize_t s3c2410_led_read(struct file *filp,char *buffer,

                                        size_t count,loff_t *ppos)

{

put_user(dev.status,(int *)buffer);  //读取LED状态

       return 1;

}

 

/*

======================================================================

s3c2410_led_write()

 描述   : 写操作函数,本实例中不做任何事

 参数   :

 返回值 : count

======================================================================

*/

static ssize_t s3c2410_led_write(struct file *filp,char *buffer,

                                   size_t count,loff_t *ppos)

{

get_user(dev.status,(int *)buffer);

if(dev.status == 0) //灭

       s3c2410_gpio_setpin(LED1, LED_OFF);

else if(dev.status == 1)//亮

     s3c2410_gpio_setpin(LED1, LED_ON);

return 1;

}

 

 

/*

======================================================================

s3c2410_led_fops

 描述   : 文件操作结构体,实现 s3c2410_button_open()等函数与open()等系统调用的连接

 参数   : 无

 返回值 : 无

======================================================================

*/

static struct file_operations s3c2410_led_fops = {

    .owner  = THIS_MODULE,

.open   = s3c2410_led_open,

.release  = s3c2410_led_release,

.ioctl    = s3c2410_led_ioctl,

.read    = s3c2410_led_read,

.write   = s3c2410_led_write,

};

 

/*

======================================================================

led_setup_cdev()

 描述   : 安装LED设备的功能函数,在设备加载模块里面调用

 参数   : 无

 返回值 : 无

======================================================================

*/

static void led_setup_cdev(void)

{

int err ,devno = MKDEV (led_major , 0);

cdev_init(&dev.cdev,&s3c2410_led_fops);

dev.cdev.owner = THIS_MODULE; 

dev.cdev.ops = &s3c2410_led_fops;  //建立设备文件操作与系统调用之间的连接

err = cdev_add(&dev.cdev,devno,1);  //向系统添加该设备

if(err)

       printk("Error %d adding LED %d",err);

}

 

/*

======================================================================

s3c2410_led_init()

 描述   : 模块加载,IO及相关变量初始化

 参数   : 无

 返回值 : 无

======================================================================

*/

static int s3c2410_led_init(void)

{

    int result;

dev_t devno = MKDEV(led_major,0);  //根据主设备号得到dev_t类型的设备号devno

if(led_major)   //如果手动分配了主设备号

       result = register_chrdev_region(devno,1,DEVICE_NAME); //向系统申请该设备号

else

{   //否则动态获取设备号

result = alloc_chrdev_region(&devno ,0 ,1,DEVICE_NAME);

led_major = MAJOR(devno);

}

if(result < 0)

return result;

led_setup_cdev();     // 注册LED设备

s3c2410_led_InitIO();    // 初始化IO端口

// initialize the vals;

dev.status = 0;    //LED的初始状态是灭

printk(DEVICE_NAME " initialized\n");

return 0;

}

 

/*

======================================================================

                          s3c2410_led_exit()

 描述   : 模块卸载函数

 参数   : 无

 返回值 : 无

======================================================================

*/

static void s3c2410_led_exit(void)

{

cdev_del(&dev.cdev);  //注销设备

unregister_chrdev_region(MKDEV(led_major,0),1);  //释放设备号

}

 

module_init(s3c2410_led_init);

module_exit(s3c2410_led_exit);

MODULE_LICENSE("GPL");  //设备许可

 

(2)编写Makefile文件(主机的/home/kernel/rootfs/rootfs/usr/led/driver目录下)

vi Makefile

在该文件中加入以下内容:(其中KDIR是内核目录,读者要根据自己的内核所在目录来设置)。

obj-m := led.o

KDIR := /home/kernel/linux-2.6.24.4

PWD := $(shell pwd)

default:

       $(MAKE) -C $(KDIR) SUBDIRS=$(PWD) modules

 或者用下面的通用形式,详细的分析参看:http://weimenlove.blog.163.com/blog/static/17775473201111945059224/

  #如果已定义KERNELRELEASE,则说明是从内核构造系统调用的,
  #因此可利用其内建语句。
  ifneq ($(KERNELRELEASE),)
          obj-m := led.o
  #要构建的模块名称为led.ko,
  #并由两个源文件生成(比如file1.c和file2.c)
          #module-objs := file1.o file2.o
  #否则,是直接从命令行调用的,
  #这时要调用内核构造系统。
  else
  KERNELDIR := /home/kernel/linux-2.6.24.4
  PWD := $(shell pwd)
  default:
          $(MAKE) -C $(KERNELDIR) M=$(PWD) modules
  endif
  clean:
          rm -rf .*.cmd *.o *.mod.c *.ko .tmp_versions *.order *symvers

(3)编译

make

编译完成后,该目录下会生成led.ko文件,该文件就是编译成功的模块文件。

(4)加载模块

首先设置内核以NFS方式加载根文件系统,在宿主机上运行minicom,打开实验箱电源,加载完U-Boot,内核和nfs文件系统之后按回车键进入目标机shell控制台,在目标机控制台中输入模块加载命令:

insmod usr/led/driver/led.ko

如果输出“led initialized”,表示led设备驱动加载成功

 

(5)编写测试文件led_test.c

进入宿主机的/home/kernel/rootfs/rootfs/usr/led/test目录,编辑测试文件led_test.c

cd /home/kernel/rootfs/rootfs/usr/led/test

vi led_test.c

在测试文件中加入如下代码:

#include

#include

#include

#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"

 

#define LED_ON 0

#define LED_OFF 1

 

int main()

{

    int fd,i,j;

int led_status = 0;  //初始状态为熄灭

    fd = open("/dev/led", O_RDWR);  //打开设备

    if (fd < 0) {  // 如果打开设备失败,退出

        perror("open device led failed !");

        exit(1);

    }

    printf("led test show. press ctrl+c to exit \n");

ioctl(fd,LED_OFF);

 

    while(1){//主循环

read (fd,&led_status,1);  //读取LED1的当前状态到led_status

if(    led_status ==0) //如果是灭的

     ioctl(fd,LED_ON);  //点亮

else if(    led_status ==1) //如果是亮的

     ioctl(fd,LED_OFF); //熄灭

       printf("led: on and off\n");

       for(i=0;i<300;i++)

                   for(j=0;j<10000;j++);

}

    close(fd);    //关闭设备

return 0;

该测试程序实现了控制S3C2410板上的灯的亮灭。

 

(6)编译测试程序

arm-linux-gcc -o led_test led_test.c

该命令的意思是使用的交叉编译器arm-linux-gcc对驱动程序led_test.c进行编译,编译成功后在test目录下会生成目标机的可执行文件led_test。

 

(7)运行测试程序

创建设备

在目标机中,通过查看/proc/devices中注册进内核的设备条目及相关的设备号。进入/dev目录,创建设备,设备名为led,属于字符型设备,主设备号是212,次设备号是0。(要与led.c文件中的定义相符):

cat /proc/devices

cd /dev

mknod led c 212 0

  

./usr/led/test/led_test或/usr/led/test/led_test

可以看到控制台会打印出一行信息:“led test show. press ctrl+c to exit ”。按“ctrl+c”可以退出测试程序。

测试程序运行时,目标机的两个led灯一亮一灭,同时,在控制台输出的信息如下:

led test show. press ctrl+c to exit                                            

led: on and off                                                              

led: on and off                                                              

led: on and off                                                              

led: on and off 

 

(8)卸载模块

rmmod /dev/led

卸载了模块以后,测试程序便不能正常运行了,如果运行会输出如下信息:“open device led: No such device or address”

       执行"insmod usr/led/driver/led.ko"时,就会调用"s3c2410_led_init"函数,执行"rmmod /dev/led"或者"rmmod led.ko"时,就会调用"s3c2410_led_exit"函数。


From: 嵌入式Linux初级实验


关键字:ARM  LED  设备驱动 引用地址:[ARM应用]LED设备驱动实例

上一篇:STM32串口下载方法
下一篇:[ARM笔记]设备驱动概述

推荐阅读最新更新时间:2024-03-16 15:21

Power Integrations推出CREE LED灯泡驱动器参考设计
美国加利福尼亚州圣何塞,2014年6月11日讯 – 高效率、高可靠性LED驱动器IC领域的世界领导者Power Integrations公司(纳斯达克股票代号:POWI)今日推出四款适用于PAR30及PAR38 LED灯泡的驱动器参考设计。这四款设计由Power Integrations和CREE共同开发而成,它们体现了Power Integrations的LYTSwitch™-4驱动器IC与CREE MT-G2 EasyWhite LED的完美结合。DER-364和DER-365分别详细介绍了低压及高压PAR 30 LED灯泡驱动器,DER-350和DER-396则分别对应于低压及高压PAR 38灯泡。这四款参考设计以及其他参考
[电源管理]
基于ARM的双频RFID读写系统设计
      摘要:本文设计了能对低频和高频RFID 卡进行操作的双频RFID 读写系统。系统基于ARM微处理器 S3C44B0X ,利用S3C44B0X 的内部接口实现与RFID 模块和其它外设的连接,简化了硬件。软件设计中,通过对ARM 嵌入式系统的编程实现对RFID 卡的读写、显示以及数据的存储、查询等功能。系统还可通过USB 接口与上位机(PC)连接,使用方便灵活。    1 引言   RFID(Radio Frequency Identification Technology ,无线射频识别技术)由于具有高速移动物体识别、多目标识别和非接触识别等特点,显示出巨大的发展潜力和应用空间,被认为是21 世纪最有发展
[网络通信]
12年LED照明市场起飞 商机优于众背光需求
LED TV市场崛起快速带动LED产业需求大增,不过LED照明市场潜力更是庞大,欧美国际大厂均看好照明商机发展,LED晶粒龙头厂晶电董事长李秉杰表示,若 LED照明应用能在2012年起飞,以每年2~5%不等的渗透率成长,将可带动LED产业再次迈向大幅成长,他指出,TV需求是所有背光需求的5倍,照明市场兴起,将可望是原有背光市场的50倍。 李秉杰表示,目前全球笔记型计算机(NB)背光源所需使用的LED数量,约仅占全球LED总产能的 30%,若是LED TV渗透率能顺利由2009年3%拉升到10%,预估全球LED总产能必须要扩充3倍才够,故市场需求如果符合预期,且各家LED厂商产能也能够各自控制,他认为,在这波LED T
[电源管理]
LED照明系统的过压过流防护
  随着LED照明灯具的逐步发展,在亮化工程辅助照明等公共场合,LED照明灯具渐渐替代了一些传统光源产品。在使用时间较长的商业应用场所,LED灯具迅速成为市场的新宠,LED行业在近些年得到了较快发展。全球LED各应用领域中,通用照明和汽车照明领域为增速最快的市场,成为推动未来LED市场规模快速增长重要驱动因素。   1.智能照明:随着LED照明产品价格持续下降,LED照明市场迅速增长。技术的进步,也使得LED照明技术从最初的简单照明向智能照明发展。智能照明不仅可以实现照明系统的智能控制,同时也成为互联网的一个入口,从而衍生更多高附加值的服务,例如健康管理、地图定位、商品导购与广告等。从传统照明进入智能照明,市场需求无疑是推动这一
[电源管理]
<font color='red'>LED</font>照明系统的过压过流防护
跨界是照明行业发展的重要趋势,LED该如何跨界?
最近,“醒狮杯”2017南海照明设计大赛正式启动。对于此次大赛主题“跨界创新·未来照明”,我是非常赞赏的。事实上,原来照明灯具应用范围较小,通过跨界,可促进 照明应用 领域向多元化发展,从而促使照明产品设计的视野更为广阔。   如何跨界呢?一、可尝试照明行业与其它行业结合:照明与医疗行业结合可推出“健康照明”、照明与农业结合可推出“植物照明”、照明与园艺结合可推出“家庭种植”等,这些都极具发展前景,也让照明应用可延伸至其它行业、领域。二、可尝试不同专业做照明设计,这些专业与照明设计具有一定相关性。目前我们学校也在做这件事情,今年的南海照明设计大赛中,我们学校组织了室内设计、景观设计、建筑设计等专业学生共同参赛,希望通过不同专业做照
[嵌入式]
LED灯管串并多少合适
我也搞了大半年的LED灯管了,经过N次试验,以T8为例,觉得0.6M的灯安排156颗灯输入功率为9W.实际用在负载上是8W左右.相同长度的T10灯因为散热面积大些,灯珠就安排到174颗,按每只0.06W计算,实际功率应该是10.44W,那么驱动功率加上损耗应该为:11.5W就可以了. 1.2M的以前都用276颗,现在有很多厂家都改为288颗,功率比以前稍大些,计算一下实际功率应该是:17.28W,取17W,加上损耗1.5W,可以把电路做到18.5W.当然功率计算要看你的灯珠质量,还要看电源的温度. 下面讲解串联和并联的关系 我们用的电源现在五花八门,市面上的降压IC最少有几十种,据我使用QX9910来说,能用在220V的电源很适
[电源管理]
Arm 全面计算解决方案重新定义视觉体验强力赋能移动游戏
Arm 全面计算解决方案重新定义视觉体验强力赋能移动游戏 新闻重点: • 全新旗舰产品 Immortalis GPU 将显著优化安卓游戏体验,并首次推出基于硬件的光线追踪功能 • 最新 Armv9 CPU 将峰值和效率性能提升至全新水平 • 新的 Arm 全面计算解决方案 (Total Compute Solutions) 可满足各级别的性能、效率和可扩展性要求,适用于各类消费级设备市场的专用处理需求 Arm® 今日宣布推出 2022 全面计算解决方案 (TCS22),可提供不同级别的性能、效率和可扩展性,以完善各类终端市场的用户体验。TCS22 的 Arm IP 组合可在一系列工作负载中实现 28% 的性能提
[传感器]
<font color='red'>Arm</font> 全面计算解决方案重新定义视觉体验强力赋能移动游戏
ARM:低功耗与高性能的结合体
几个小时的电池时间几乎是所有笔记本电脑使用者的烦恼。那么你曾经想过会有十几小时甚至更长使用时间的笔记本吗?戴尔不久前推出一款型号为Latitude E6400的“混合”笔记本电脑,宣称具有长达19个小时的电池使用时间,原因是使用了基于ARM的处理器。 生活中见惯了品牌繁杂、形态万千的手机,但你知道目前世界上平均每部新手机上就有2颗ARM处理器吗?其实不光是笔记本、手机,还有汽车、机顶盒等很多与人类生活息息相关的嵌入式应用产品中都会存在一个共同的名字——ARM。 ARM是什么?ARM(Advanced RISC Machines),既可以认为是一个公司的名字,也可以认为是对一类处理器的通称,还可以认为是一个技术
[嵌入式]
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved