1.概述

一个软件系统分为:应用程序、库、操作系统、驱动程序。

(1)应用程序使用库函数提供的open打开LED设备文件。

(2)库根据open函数传入的参数执行“swi”指令,引起CPU异常进入内核。

(3)内核的异常处理函数根据这些参数找到对应的驱动程序,并且将打开的设备文件句柄返回给库,进而返回给应用程序。

(4)应用程序获得句柄后,使用库提供的write或ioclt函数发出控制命令。

(5)库根据write或ioclt函数传入的参数执行“swi”指令,引起CPU异常,进入内核。

(6)内核的异常处理函数根据参数调用相关驱动程序,点亮LED灯。

一般来说,当应用程序调用open、read、write、ioctl、mmap等函数后,系统将会调用驱动程序中的open、read、write、ioctl、mmap等函数进行相关操作。

驱动程序通过静态链接或者动态加载的方式编进内核。

2.Linux驱动程序

Linux的外设可以分为3类:字符设备(character device)、块设备(block device)和网络接口(network interface)。

字符设备是能够向字节流一样访问的设备。比如串口、LED等等。应用程序可以通过设备文件来访问字符设备(/dev/ttySAC0)。

块设备上的数据以块的形式存放,比如NAND Flash上的数据以页为单位存放。

网络接口具有字符设备、块设备的部分特点。

3.开发步骤

(1)查看原理图,数据手册,了解设备的操作方法。

(2)在内核中找到相近的驱动程序,以它为模板进行开发。

(3)编写内核的初始化程序,比如向内核注册驱动程序、卸载驱动程序。

(4)设计所要实现的操作,比如open、close、read、write等函数。

(5)编译驱动程序到内核中,后者动态加载进内核。

(6)测试驱动程序。

static struct class *firstdrv_class;

static struct class_device*firstdrv_class_dev;

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

{

printk("first_drv_open\n");

return 0;

}

static ssize_t leddrv_write(struct file *file, const char __user *buf, size_t count, loff_t * ppos)

{

int val;

printk("first_drv_write\n");

return 0;

}

static struct file_operations leddrv_fops = {

.owner  =   THIS_MODULE,    /* 这是一个宏__this_module?? */

.open   =   leddrv_open,

.write = leddrv_write,

};

int major;

static int leddrv_init(void)

{

major = register_chrdev(0, "led_drv", &first_drv_fops); // 自动分配一个主设备号

firstdrv_class = class_create(THIS_MODULE, "firstdrv");

firstdrv_class_dev = class_device_create(firstdrv_class, NULL, MKDEV(major, 0), NULL, "xyz");            /*自动创建 /dev/xyz */

return 0;

}

static void leddrv_exit(void)

{

unregister_chrdev(major, "first_drv"); //

class_device_unregister(firstdrv_class_dev);

class_destroy(firstdrv_class);

}

module_init(leddrv_init);

module_exit(leddrv_exit);

Makefile文件

KERN_DIR = /work/system/linux-2.6.22.6

all:

make -C $(KERN_DIR) M=`pwd` modules

clean:

make -C $(KERN_DIR) M=`pwd` modules clean

rm -rf modules.order

obj-m += led_drv.o

注:需要仔细考虑驱动程序中需要包括哪些头文件!

cat  /proc/devices查看注册进内核的驱动程序。

在当前文件目录下进行make modules,生成驱动模块led_drv.ko,将他放到单板根文件系统的/lib/modules/2.622.6/目录下,然后“insmod led_drv.ko”命令进行加装进内核,“rmmod led_drv.ko”为卸载命令。

3.应用程序

#include<sys/types.h>

#include<sys/stat.h>

#include<fcntl.h>

#include<stdio.h>

int main(int argc, char **argv)

{

int fd;

int val =1;

fd =open("/dev/xxx",O_RDWR);

if(fd<0)

printf("can't open!\n");

write(fd,&val,4);

return 0;

}

arm-linux-gcc -o ledtest ledtest.c编译应用程序。

4.创建设备节点

直接运行测试文件输出openfailed。原因是没有创建设备节点。

进行测试,将ledtest 拷贝到网络文件系统下,开发板以网络文件系统方式启动,运行该文件。

测试结果:

(1)手动在单板根文件系统下建立设备节点:

mknod  /dev/xxx  c  231  0

进行测试,将ledtest 拷贝到网络文件系统下,开发板以网络文件系统方式启动,运行该文件。

测试结果:

(2)利用mdev自动建立设备节点

在驱动程序中添加代码创建类和设备,使得在/sys/目录下生成驱动程序信息,mdev机制会根据该信息,在/dev/目录下自动创建设备节点。

在/dev/init.d/rcS脚本文件中我们已经设置了mdev机制。

我们的脚本文件:

韦东山第12课2.2节总结:韦东山老师的视频加载驱动后,发现这些函数未识别即unknown symbol的错误。加上MODULE_LICENSE("GPL")。但是我在编译的时候还是出现了警告,不过程序可以成功测试。

测试结果如下:

(a)查看/dev/first_drv发现已经存在,系统自动帮我们创建了字符设备节点。

(b)查看sys目录下的文件信息。

韦东山第12课2.3节总结:

7.完善硬件的操作:

1)看原理图:确定物理连接的端口。

2)看芯片手册:确定需要操作的寄存器。0x21180000

3)写代码:单片机实验时是直接操作物理地址;而驱动程序中是操作虚拟地址,需要经过映射ioremap()。

下面我们一步一步往下走:

1)看JZ2440原理图

这三个灯分别接到了GPF4、GPF5、GPF6三个引脚。

1)查2440手册确定寄存器

需要操作GPFCON、GPFDAT两个寄存器,GPFCON的寄存器地址为0x56000050,GPFDAT的寄存器地址为0x56000054。

volatile unsigned long *gpfcon = NULL;

volatile unsigned long *gpfdat = NULL;

gpfcon = (volatile unsigned long *)ioremap(0x56000050, 16); //地址映射

gpfdat = gpfcon + 1;

static int first_drv_open(struct inode *inode, struct file *file)函数中配置寄存器

/* 配置GPF4、5、6为输出*/

*gpfcon &= ~((0x3<<(4*2)) | (0x3<<(5*2)) | (0x3<<(6*2)));

*gpfcon |= ((0x1<<(4*2)) | (0x1<<(5*2)) | (0x1<<(6*2)));

static ssize_t first_drv_write(struct file *file, const char __user *buf, size_t count, loff_t * ppos)函数中点灯

int val;

copy_from_user(&val, buf, count); // copy_to_user();将buf拷贝到内核空间

if (val == 1)

{

// 点灯

*gpfdat &= ~((1<<4) | (1<<5) | (1<<6));

}

else

{

// 灭灯

*gpfdat |= (1<<4) | (1<<5) | (1<<6);

}

测试程序修改:

#include <sys/types.h>

#include <sys/stat.h>

#include <fcntl.h>

#include <stdio.h>

/* firstdrvtest on

* firstdrvtest off

*/

int main(int argc, char **argv)  //argc为参数个数 ,argv[1]为第二个参数

{

int fd;

int val = 1;

fd = open("/dev/xyz", O_RDWR);

if (fd < 0)

{

printf("can't open!\n");

}

if (argc != 2)

{

printf("Usage :\n");

printf("%s <on|off>\n", argv[0]);

return 0;

}

if (strcmp(argv[1], "on") == 0)

{

val  = 1;

}

else

{

val = 0;

}

write(fd, &val, 4);

return 0;

}

运行测试程序./1st_test on灯全亮,./1st_test off灯全灭。

进一步完善程序:(达到指定哪盏灯亮)

具体点亮哪个灯:利用次设备号来实现。

驱动需要修改的代码如下:

static int first_drv_init(void)

{

int i;

char * led[3]={"led0","led1","led2"}

major=register_chrdev(0,"first_drv",&first_drv_fop);      //

firstdrv_class = class_create(THIS_MODULE,"firstdrv");

if(IS_ERR(firstdrv_class))

return PTR_ERR(firstdrv_class);

firstdrv_class_dev[0]=class_device_create(firstdrv_class,NULL,MKDEV(major,0),NULL,"leds");

for(i=1;i<4;i++)

firstdrv_class_dev[i]==class_device_create(firstdrv_class,NULL,MKDEV(major,i),NULL,led[i-1]);

if(unlikely(IS_ERR(firstdrv_class_dev[0])))

return PTR_ERR(firstdrv_class_dev[0]);

gpfcon=(unsigned long *)ioremap(0x56000050,16);

gpfdat = gpfcon+1;

return 0;

}

static ssize_t first_drv_write (struct file *file, const char __user *buf, size_t count, loff_t *lof)

{

int val;

printk("first_drv_write\n");

val=*buf;

printk("buf=%d",val);

int minor = MINOR(file->f_dentry->d_inode->i_rdev);

copy_from_user(&val, buf, count);  //copy_to_user();

switch(minor)

case 0:

{

//s3c2410_gpio_cfgpin(S3C2410_GPF4, S3C2410_GPF4_OUTP);

s3c2410_gpio_setpin(S3C2410_GPF4, (val & 0x1));

s3c2410_gpio_setpin(S3C2410_GPF5, (val & 0x1));

s3c2410_gpio_setpin(S3C2410_GPF6, (val & 0x1));

break;

}

case 1:

{

s3c2410_gpio_setpin(S3C2410_GPF4, (val & 0x1));

break;

}

case 2:

{

s3c2410_gpio_setpin(S3C2410_GPF5, (val & 0x1));

break;

}

case 3:

{

s3c2410_gpio_setpin(S3C2410_GPF6, (val & 0x1));

break;

}

return 0;

}

测试程序修改后的代码如下:

#include <sys/types.h>

#include <sys/stat.h>

#include <fcntl.h>

#include <stdio.h>

/*

*  ledtest <dev> <on|off>

*/

void print_usage(char *file)

{

printf("Usage:\n");

printf("%s <dev> <on|off>\n",file);

printf("eg. \n");

printf("%s /dev/leds on\n", file);

printf("%s /dev/leds off\n", file);

printf("%s /dev/led1 on\n", file);

printf("%s /dev/led1 off\n", file);

}

int main(int argc, char **argv)

{

int fd;

char* filename;

char val;

if (argc != 3)

{

print_usage(argv[0]);

return 0;

}

filename = argv[1];

fd = open(filename, O_RDWR);

if (fd < 0)

{

printf("error, can't open %s\n", filename);

return 0;

}

if (!strcmp("on", argv[2]))

{

// ÁÁµÆ

val = 0;

write(fd, &val, 1);

}

else if (!strcmp("off", argv[2]))

{

// ÃðµÆ

val = 1;

write(fd, &val, 1);

}

else

{

print_usage(argv[0]);

return 0;

}

return 0;

}

第一个驱动程序(点亮LED灯)相关推荐

  1. 单片机第一个程序----点亮LED灯

    单片机第一个程序----点亮LED灯 准备: 安装Keil软件 软件官网 单片机实验板(某宝可以买到) 简单的C语言编程能力 单片机C语言教程 简单介绍下keil软件的使用方法: 第一步:新建工程,这 ...

  2. 使用驱动程序点亮LED灯

    继第一节第一个驱动程序框架记录之后,本篇文章将会在上一篇驱动程序的框架下编写点亮LED的驱动程序,同样会对上一个框架进行修改,优化.接下来进入正题 1.点亮LED程序框架分析 在最开始之前先来梳理一下 ...

  3. linux开发板led怎么亮,飞凌OK6410开发板(裸板) 第一个点亮LED灯程序

    飞凌OK6410开发板(裸板) 第一个点亮LED灯程序,主要的C程序,完整程序请下载附件. #define rGPMCON          (*(volatile unsigned *)(0x7F0 ...

  4. 第一节:C#工业控制编程基础--点亮LED灯实验

    第一节:C#工业控制编程基础–点亮LED灯实验 文章目录 第一节:C#工业控制编程基础--点亮LED灯实验 一.实验目的: C#入门基础学习. 二.实验内容: 用C#控制LED的亮灭. 三.实验步骤: ...

  5. STM32笔记2-使用库函数点亮LED灯

    目录 一.硬件电路设计 二.软件设计 1.工程配置 2.程序编写 (1)led头文件 (2)Led_Init()函数编写 三.实验测试 四.使用宏定义 1.修改后的led.h中程序 2.修改后的Led ...

  6. STC89C52单片机 点亮LED灯

    点亮LED灯 一.什么是LED LED全称为半导体发光二极管,采用半导体材料制成的,以直接将电能转化为光能,电号转换成光信号的发光器件:其特点是功耗低.高亮度.色彩艳丽.坑振动.寿命长(正常发光8-1 ...

  7. 【STM32学习笔记-点亮LED灯】

    STM32学习笔记-点亮LED灯 文章目录 STM32学习笔记-点亮LED灯 一.原理图分析 二.代码分析 1.mian函数 2.led.c函数 3.led.h函数 4.函数文件整理 5.LED_In ...

  8. S3C2440之裸机之C语言按键点亮LED灯

    虚拟机环境:Oracle VM VirtualBox Linux系统:ubuntu_14.04.6 交叉编译工具:[100ask分享的所有文件](https://eyun.baidu.com/s/3b ...

  9. 使用STM32f103点亮led灯——库函数版本

    本文章是结合我这一年的学习与应用,总结出来的经验与知识(主要为了应用),欢迎读者们学习和指导. 前言 资料 : 链接:https://pan.baidu.com/s/1pr57NSXFmax06kqP ...

最新文章

  1. wdcp php5.3 pdo_mysql,WDCP常用组件(memcache、mysqli、PDO_MYSQL、mysql innodb、libmcrypt、php zip)的安装方法...
  2. cocos2d-x CCParticleSystem粒子系统
  3. hdu 4891 模拟
  4. 各种浏览器的userAgent
  5. 规格选择_止水螺杆规格及选择
  6. N551JM集显和独显切换
  7. Java泛型面试问题
  8. 花书+吴恩达深度学习(十四)卷积神经网络 CNN 之经典案例(LetNet-5, AlexNet, VGG-16, ResNet, Inception Network)
  9. 实验4-1-6 求分数序列前N项和 (15 分)
  10. jQuery Mobile中可折叠块collapsed的data-*选项
  11. 前端ajax怎么样遍历list_五大前端小白入门时最容易掉的坑,可得提防点!
  12. 佛言:人有二十难(为)
  13. Spring-IOC本质
  14. allure趋势图无数据
  15. Liunx 安装redis
  16. 计算机被格式化怎么找回资料,文件被格式化 硬盘格式化删除的文件怎么找回...
  17. 微服务模式笔记:服务分解策略
  18. 励志长篇小说《周兴和》书连载之二饥饿寒冷童年
  19. Aspx.Net的Aspx页面和Aspx.cs联用
  20. 一键卸载Ubuntu的火狐浏览器

热门文章

  1. ST001 | 移动 App 应用测试方法与思路
  2. 【535. TinyURL 的加密与解密】
  3. 哈希冲突和哈希冲突攻击解析
  4. 解放号测试家亮相西安2016“全国大众创业万众创新活动周”
  5. 普通的Shader-物体移动身后残影的一种实现
  6. 多个优秀的OCR算法解读
  7. 谷歌浏览器登录不上 sarai浏览器可以登录的问题
  8. Android开发社交软件类应用时的若干问题
  9. 【课程】并行计算导论
  10. Anchor DETR