3.修改第一个程序来点亮LED

在上一节中已经将驱动程序框架搭建好了

接下来开始写硬件的操作(控制LED):

(1)看原理图,确定引脚

(2)看2440手册

(3)写代码(需要使用ioremap()函数映射虚拟地址,在linux中只能使用虚拟地址)

(4)修改上一节的测试程序

(5)使用次设备号来控制设备下不同的灯

1.看led引脚

最终确定: LED1 ->GPF4  LED2 ->GPF5   LED3 ->GPF6

2.看2440手册

配置GPFCON[15:0](0x56000050)的位[8:9]、位[10:11]、位[12:13] 都等于0x01(输出模式)

控制GPFDAT[7:0](0x56000054)中的位4~6来使灯亮灭(低电平亮)

3.写代码

3.1添加全局变量:

volatile unsigned long *GPFcon=NULL;       

volatile unsigned long *GPFdat=NULL;

3.2 first_drv_init入口函数中使用ioremap()映射虚拟地址:

GPFcon = ioremap(0x56000050, 16);   //ioremap:物理地址映射,返回虚拟地址

GPFdat=GPFcon+1;             //long:32位,所以GPFdat=0x56000050+(32/8)

3.3 first_drv_exit出口函数中注销虚拟地址:

iounmap(GPFcon);          //注销虚拟地址

3.4 first_drv_open函数中添加配置GPFCON:

*GPFcon&=~ ((0X11<<8)| (0X11<<10)| (0X11<<12)); 

*GPFcon|=    ((0X01<<8)| (0X01<<10)| (0X01<<12)); 

3.5 first_drv_write函数中添加拷贝应用层数据,然后来控制GPFDAT:

/*copy_to_user():将数据上给用户*/
copy_from_user(&val,buf,count);      //从用户(应用层)拷贝数据
 if(val==1)                  //点灯(低电平亮)
 {
      *GPFdat&=~((0X1<<4)| (0X1<<5)| (0X1<<6));
 }
 else                  //灭灯
 {
     *GPFdat|=((0X1<<4)| (0X1<<5)| (0X1<<6));
 }

4.修改测试程序main()

代码如下:

int main(int argc,char **argv) //argc:参数个数,argv数组
{
int fd1, fd2;
int val=1;
fd1 = open("/dev/xyz",O_RDWR);  //打开/dev/xxx设备节点
if(fd1<0)                   //无法打开,返回-1
  printf("can‘t open%d!\n", fd1);
   if(argc!=2)
     {
            printf("Usage:\n");
            printf("%s <on|off>",argv[0]);
            return 0;
     }

  if(strcmp(argv[1],"on")==0)   //开灯
      {
          printf("led on...\n");
          val=1;
      }
  else                         //关灯
      {
          printf("led off...\n");
          val=0;
      }

write(fd1, &val, 4);
return 0;
}

当输入first_driver_text on点3个灯, 否则关3个灯

若参数不等于2时,不能控制点灯

如果我们想分别控制不同的灯,该怎么做?

可以使用此设备号,此设备号就是用来区分同一设备下不同子设备

5使用次设备号来控制设备下不同的灯

我们先来看下面两个函数MAJOR和MINOR,分别是提取主次设备号

minor=MINOR(inode->i_rdev);    //open函数中提取次设备号
major=MAJOR(inode->i_rdev);    //open函数中提取主设备号

minor=MINOR (file->f_dentry->d_inode->i_rdev);  //write/read函数中提取次设备号
major= MAJOR (file->f_dentry->d_inode->i_rdev); //write/read函数中提取主设备号

思路如下:

在测试程序中:

通过dev[1]来open打开不同的子设备节点,然后通过dev[2]来write写入数据

实例: first_driver_text led1 on        //点亮led1

在first_dev.c驱动文件中:

first_drv_init函数中创建不同的子设备节点

first_drv_exti函数中注销不同的子设备节点

first_drv_open函数中通过MINOR(inode->i_rdev)来初始化不同的灯

first_drv_write函数中通过MINOR(file->f_dentry->d_inode->i_rdev)来控制不同的灯

如下图,insmod后自动注册3个设备节点

测试程序如下:

#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;
}

驱动程序如下:

#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/fs.h>
#include <linux/init.h>
#include <linux/delay.h>
#include <asm/irq.h>#include <asm/arch/regs-gpio.h>
#include <asm/hardware.h>
#include <asm/uaccess.h>
#include <asm/io.h>

static struct class *firstdrv_class;               //创建一个class类
static struct class_device   *firstdrv_class_devs[4]; //创建类的设备,led,led1,led2,led3

volatile unsigned long *GPFcon=NULL;
volatile unsigned long *GPFdat=NULL;

/*1写出驱动程序first_drv_open first_drv_write */
static int first_drv_open(struct inode *inode, struct file  *file)
{
  int minor=MINOR(inode->i_rdev);
  printk("first_drv_open\n");      //打印,在内核中打印只能用printk()
  GPFcon = ioremap(0x56000050, 16);   //ioremap:物理地址映射,返回虚拟地址
  GPFdat=GPFcon+1;                   //long:32位,所以GPFdat=0x56000050+(32/8)
       switch(minor)
   {
   case 0:                              //进入led设备,控制所有led
   *GPFcon&=~ ((0X3<<8)| (0X3<<10)| (0X3<<12));
   *GPFcon|=    ((0X01<<8)| (0X01<<10)| (0X01<<12));
        break;

   case 1:                              //进入led1设备,控制 led1
   *GPFcon&=~ ((0X3<<8) );
   *GPFcon|=    (0X1<<8) ;
        break;     

   case 2:                                                 //进入led2设备,控制 led2
  *GPFcon&=~ ((0X3<<10) );
  *GPFcon|=  (0X1<<10) ;
         break;

   case 3:                              //进入led3设备,控制 led3
   *GPFcon&=~ ((0X3<<12) );
   *GPFcon|=    ((0X1<<12) );
        break;
   }
   return 0;

}

/*参数filp为目标文件结构体指针,buffer为要写入文件的信息缓冲区,count为要写入信息的长度,ppos为当前的偏移位置,这个值通常是用来判断写文件是否越界*/
static ssize_t first_drv_write(struct file *file, const char __user *buf, size_t count, loff_t * ppos)
{
       int val;
       int minor=MINOR(file->f_dentry->d_inode->i_rdev);
       copy_from_user(&val,buf,count);      //通过用户(应用层)拷贝数据
       switch(minor)
        {
        case 0:                                               //进入led设备,控制所有led
         printk("led0,%d\n",val);

        if(val)       //开灯
        {*GPFdat&=~ ((0X1<<4)| (0X1<<5)| (0X1<<6));
        *GPFdat|=      ((0X0<<4)| (0X0<<5)| (0X0<<6));   }
        else     //关灯
          {*GPFdat&=~ ((0X1<<4)| (0X1<<5)| (0X1<<6));
          *GPFdat|=      ((0X1<<4)| (0X1<<5)| (0X1<<6));   }
                break;

        case 1:                                               //进入led1设备,控制 led1
              printk("led1,%d\n",val);
               if(val)      //开灯
                     {*GPFdat&=~ (0X1<<4);
                      *GPFdat|=      (0X0<<4);  }
                      else         //关灯
                     {  *GPFdat&=~  (0X1<<4);
                             *GPFdat|=      (0X1<<4);  }
                break;

        case 2:                                         //进入led2设备,控制 led2
               printk("led2,%d\n",val);
               if(val)      //开灯
                      {*GPFdat&=~ (0X1<<5);
                      *GPFdat|=      (0X0<<5);  }
                      else         //关灯
                             {*GPFdat&=~  (0X1<<5);
                             *GPFdat|=      (0X1<<5);  }
              break;

        case 3:                                               //进入led3设备,控制 led3
              printk("led3,%d\n",val);
               if(val)      //开灯
                      {*GPFdat&=~ (0X1<<6);
                      *GPFdat|=      ( 0X0<<6); }
                      else         //关灯
                             {*GPFdat&=~ (0X1<<6);
                             *GPFdat|=      (0X1<<6);  }
                break;
        }
     return 0;
}

/*2定义file_operations结构体来封装驱动函数first_drv_open first_drv_write */
 static struct file_operations first_drv_fops = {
    .owner  =   THIS_MODULE,     //被使用时阻止模块被卸载
    .open   =   first_drv_open,
    .write   =   first_drv_write,
  };

int major;              //定义一个全局变量,用来保存主设备号
int first_drv_init(void)
{
    int i;
   /*3 register_chrdev注册字符设备*/
  /*如果设置major为0,表示由内核动态分配主设备号,函数的返回值是主设备号*/
  major=register_chrdev (0, "first_drv", &first_drv_fops);
  firstdrv_class= class_create(THIS_MODULE,"firstdrv");
//创建类,它会在sys目录下创建firstdrv这个类  

  firstdrv_class_devs[0]=class_device_create(firstdrv_class,NULL,MKDEV(major,0),NULL,"led");
//创建类设备,它会在firstdrv_class类下创建led设备,然后mdev通过这个自动创建/dev/xyz这个设备节点

  for(i=1;i<4;i++)   //创建led1 led2 led3 设备节点,控制led1 led2 led3
 {
 firstdrv_class_devs[i]=class_device_create(firstdrv_class,NULL,MKDEV(major,i),NULL,"led%d",i);
 }
  return 0;
}

/*6 写first_drv_exit出口函数*/
void first_drv_exit(void)
{
 int i;
 unregister_chrdev (major, "first_drv");  //卸载驱动,只需要主设备号和设备名就行
 class_destroy(firstdrv_class);                      //注销类,与class_create对应

  for(i=0;i<4;i++)                              //注销类设备led,led1,led2,led3
  class_device_unregister(firstdrv_class_devs[i]);
    iounmap(GPFcon);          //注销虚拟地址
}

/*5 module_init修饰入口函数*/
module_init(first_drv_init);

/*7 module_exit修饰出口函数*/
module_exit(first_drv_exit); 

MODULE_LICENSE("GPL v2");  //声明许可证
时间: 2024-10-09 08:02:05

3.修改第一个程序来点亮LED的相关文章

S5PV210裸机程序之点亮LED【基于九鼎X2103BVS】

前期准备: 查阅x210bv3s.pdf得到开发板LED模块的原理图: 从原理图中了解到:其中三颗LED分别接在GPJ0_3.GPJ0_4.GPJ0_5,还有一颗LED接在PWMTOUT1.下面我们继续查阅手册 查阅x210cv3.pdf得到PWMTOUT1对应的引脚原理图: 由此我们得到了PWMTOUT1所对应的引脚为GPD0_1. 3.  查阅S5PV210_UM_REV1.1.pdf手册,得到引脚对应的寄存器地址以及相关寄存器的设置: 从手册中,我们可知一下三个重要的参数: 1. GPJ0

Mini2440上的第一个程序——点亮Led

手头的Mini2440搁置了两年半之后,我再次决定拿出它,重新尝试嵌入式Linux的学习. 我使用的是友善之臂的Mini2440开发板.韦东山的<嵌入式Linux应用开发完成手册>及其视频教程.所以,本篇文章中所涉及到的各种软件均可在以下两处找到: Mini2440开发板的配套光盘 韦东山JZ2440开发板的光盘 JZ2440是韦东山出品的开发板,作为<嵌入式Linux应用开发完全手册>的配套硬件,它和Mini2440相差无几,所以我这里用的是Mini2440. 一.目标 动手之前

java学习 之 第一个程序及认识

以前也看过一系列的java方面的程序,但是还没有正式敲过,今天正式学习并且正式敲出代码.在这里记录下来今日所得 写作工具:Notepad++ 在写作工具方面好多人建议用 记事本,但是我还是认为用 Notepad++ 这个编辑工具比较好. 这个相较于记事本来说,关键字可以编写,能够提高学习中的拼写错误提交效果: 再者,写好以后也要手动执行命令,更能够知道程序是怎么运行的. 第一个程序 public class HelloWorld { public static void main(String

java基础--JDK安装、环境变量配置、工具开发第一个程序、数据类型、运算符

**-----Java基础大纲-----**   **-----本章节-----** 1.Java语言的历史.特点及工作原理 2.JRE和JDK的介绍 3.Java运行环境和开发工具 4.Java基础语法 **-----下一章节-----** 5.条件语句 6.循环 7.数组 ============================================== 一:历史及开发准备 1.Java发展历程及来源 (1)发展历程 1996年1月,Sun公司发布了Java的第一个开发工具包(JD

在Win7(64位)使用VS2015运行《OpenGL编程指南》第八版第一章程序的方法

前言:笔者第一次用vs2015来实现<OpenGL编程指南>第八版第一个程序时确实花费了不少时间,按照网上教程,尝试了各种方法,最终花费了两个上午加一个下午的时间, 成功运行了程序,花了这么多时间,确实让人懊恼,现在把运行程序的步骤记录下来,以便查阅. 1.第一步,下载oglpg-8th-edith. 如果去书本上的官网下载,下载的是第九版的,而不是第八版的源码. 去其他网站下载,下载的这个包里面没有第一章的源码,可以网上黏贴其他人的代码,建议下第八版源码,下载网址:链接:http://pan

STM32F407第一步之点亮LED

STM32F407第一步之点亮LED. 要点亮LED,首先了解一下F4的GPIO模块.首先看一下STM32F4数据手册,GPIO模块的内部结构图 看上去有点复杂,不要怕,慢慢理解就可以了.对外引脚那里二极管就是保护的作用.通过上.下拉对应的开关配置,控制引脚默认状态的电压,开启上拉的时候引脚电压为高电平,开启下拉的时候引脚电压为低电平,这样可以消除引脚不定状态的影响.但是这个不应该用来作为外部的上拉或下拉用,如按键的拉电阻不能用这个内部来作用,如果用可能会引起按键不稳定. GPIO 具有了“推挽

arm9 点亮led程序优化导致全亮问题

如需转载请注明出处 本实验是arm9 裸板程序,主要功能是循环点亮4个led. 参考伟山东的<嵌入式linux应用开发>点亮led节. 电路图如下: 代码如下: head.s .text .global _start _start: ldr r0, =0x56000010 @ WATCHDOG寄存器地址 mov r1, #0x0 str r1, [r0] @ 写入0,禁止WATCHDOG,否则CPU会不断重启 ldr sp, =1024*4 @ 设置堆栈,注意:不能大于4k, 因为现在可用的内

起航,第一个程序——还是LED灯

如同学基本语言一样,helloworld是很多语言的第一个程序.在嵌入式开发中,点亮LED灯也是各种架构和开发板的第一个程序,其中很多东西是和单片机例如stm32是类似的,只是,现在我们没有了库函数,我们要自己完成一些东西. 先说启动文件,st官方已结给我们做好了,但是jz2440开发板没有统一的启动文件,需要自己编写,那么,基础的arm汇编就得有所熟悉,在之后的学习中,遇到一个指令就学习一个. (汇编)指令是CPU机器指令的助记符,经过编译后会得到一串1.0组成的机器码,可以由CPU读取执行.

2.1 我们的第一个程序

1.       控制台应用程序. 在我们这个培训中主要使用控制台应用程序来讲解知识点和做练习. 什么是控制台程序? 控制台程序运行在dos窗口.没有可视化的界面.可以通过Dos窗口进入输入和输出显示. 为什么使用控制台程序作为编程的入门? 控制台程序没有复杂的图形界面,非常简单.可以让初学者专注于编程的基本功学习. 2.       创建我们的第一个控制台程序 A) 文件->新建->项目,在模板中选择visual c#,然后在右边选择控制台应用程序. B) 修改解决方案名称为MyStudyS