uClinux学习笔记之三  uClinux驱动开发初步

 

uClinuxlinux的驱动开发是一致的,只是调试的方式不一样,学习uClinux的驱动开发也就是学习linux驱动开发的过程。linux驱动的调试需要至少一台电脑,外加像GDBKDBG这样的源码级的内核调试工具,如果装一个虚拟机的话,那么只需要一台电脑即可了,但这样对电脑的性能有更高的要求。对于像uClinux的驱动调试,则最好是有开发板,也可以用模拟器。在ARM芯片上的移植是uClinux的一个重要应用。而最常用的仿真ARM的模拟器有国人开发的开源软件skyeye,不仅可以仿真ARM内核,并对以ARM作为内核的芯片作了进一步的支持。可以用skyeye -h 开查看skyeye能仿真的ARM芯片。当然,也可以对skyeye作一些扩展以满足自己的要求。以下是如何在uClinux添加驱动的过程。

一、编写一个简单的字符驱动程序

//---------------------------------------------mydevice.c-------------------------------------------

#include <linux/module.h>

#include <linux/init.h>

#include <linux/fs.h>

#include <asm/uaccess.h>

MODULE_LICENSE("GPL");

#define MAJOR_NUM 254 //主设备号

#define DRIVER_NAME "mydevice"

static ssize_t mydevice_read(struct file *, char *, size_t, loff_t*);

static ssize_t mydevice_write(struct file *, const char *, size_t, loff_t*);

//初始化字符设备驱动的file_operations结构体

struct file_operations mydevice_fops =

{

      .read = mydevice_read,

      .write = mydevice_write,

};

static int global_var = 0; //"mydevice"设备的全局变量

 

int  mydevice_init(void)

{

      int ret;

      //注册设备驱动

      printk("<0> mydevice_init /n");

      ret = register_chrdev(MAJOR_NUM,DRIVER_NAME,&mydevice_fops);

      printk("<0> register_chr_dev return %d /n",ret);

      if (ret < 0)

      {

           printk("<0>mydevice register failed /n");

      }

      else

      {

           printk("<0>mydevice register success /n");

      }

      return ret;

}

void mydevice_exit(void)

{

      printk("<0>mydevice_exit!/n");

      unregister_chrdev(MAJOR_NUM,DRIVER_NAME);

}

static ssize_t mydevice_read(struct file *filp,char *buf,size_t len,loff_t *off)

{

      //copy global_var from kernel space to user space

      if(copy_to_user(buf,&global_var,sizeof(int)))

      {

           return -EFAULT;

      }

      return sizeof(int);

}

static ssize_t mydevice_write(struct file *filp,const char *buf,size_t len,loff_t *off)

{

      //copy data from user sapce to kernel space

      if(copy_from_user(&global_var,buf,sizeof(int)))

      {

           return -EFAULT;

      }

      return sizeof(int);

}

//-----------------------------------------------------------------------------------------------------------------

二、将驱动静态编译进uClinux内核

要将一个写好的驱动加入uClinux,需要对配置文件和Makefile作一些修改,由于我们的这个驱动程序是作为一个字符型驱动程序,我们把mydevice.c拷到(uClinux目录)/(linux 内核目录)/drivers/char目录下。并对这个目录下的Config.inMakefile以及mem.c作一些修改。

1、         修改Config.in:

comment  'Character devices' 这行底下加上以下一行:

bool  'support for mydevice'  CONFIG_MYDEVICE  y

2、         修改Makefile

#

# uClinux drivers

#

下加上以下内容:

obj-$(CONFIG_MYDEVICE) += mydevice.o

3、         修改mem.c

mem.c主要是初始化一些以虚拟设备,这些设备通常是以内存作为基础“设备”的,我们把mydevice的初始化代码加入其中:

mem.c文件的起始位置加上以下代码:

#ifdef CONFIG_MYDEVICE

extern int mydevice_init(void);

#endif

修改mem.cchr_dev_init函数    

if (devfs_register_chrdev(MEM_MAJOR,"mem",&memory_fops))

       printk("unable to get major %d for memory devs/n", MEM_MAJOR);

  memory_devfs_register();

  rand_initialize();

#ifdef CONFIG_MYDEVICE

  mydevice_init();

#endif

#ifdef CONFIG_I2C

  i2c_init_all();

#endif

 

4、         编译mydevice驱动

所有要作的修改就是这么简单,接下来的问题就是怎样将我们的驱动编译进内核了

(1) 配置内核

运行make menuconfig

进入Kernel/Library/Defaults Selection菜单

选上Customize Kernel Settings

退出并保存设置

在新出现的菜单中进入Character devices子菜单

选上support for mydevice

退出并保存设置

make depmake命令生成内核镜像和内存文件系统镜像

 

三、测试我们的驱动

      如何与我们的驱动打交道呢,答案当然是建立一个应用程序了,在《学习笔记之二》中已经详细讲解了如何将一个应用程序加入uClinux,现在我们就来看看怎样在应用程序里调用我们的驱动。

   为了让我们驱动能够为应用程序所用,必须事先创建一个文件结点,在通用linux 系统中我们可以用mknod命令,建立的结点会出现在/dev目录下。但是在嵌入式linux系统uClinux中,更好的办法是让uClinux在启动时为我们做这些事:

打开(uClinux目录)/vendors/GDB/ARMulator-EB目录下的Makefile文件,作些修改,如下:

(注:请根据配置时在Vendor/Product Selection菜单里的选择来选择vendors下的相应Makefile)

DEVICES = /

      mydevice,c,254,0 /

      tty,c,5,0      console,c,5,1      cua0,c,5,64      cua1,c,5,65  /

其中mydevice,c,254,0就是我们添加的内容,各项的含义分另是:

mydevice  :设备名

c                :字符设备

254           :主设备号

0                           :副设备号

 

 

下面来看看我们的测试程序:

//----------------------------------hello.c--------------------------------------

#include <sys/types.h>

#include <sys/stat.h>

#include <stdio.h>

#include <fcntl.h>

int main(void)

{

      int fd,num;

      fd = open("/dev/mydevice",O_RDWR,S_IRUSR | S_IWUSR);

      if(fd != -1 )

      {

           read(fd,&num,sizeof(int));

           printf("The globalvar is %d /n",num);

           printf("please input the num written to globalvar /n");

           scanf("%d",&num);

           write(fd,&num,sizeof(int));

           read(fd,&num,sizeof(int));

           printf("the globalvar is %d /n",num);

           close(fd);

      }

      else

      {

           printf("device open failure /n");

      }

}

这段代码应该很好理解,首先打开设备,然后写入再读出来

配置好后重新编译内核,用skyeye仿真

 

 

 

到此,我们已经开始uClinux驱动开发之旅了

 

参考资料:

http://dev.yesky.com/186/2623186.shtml

Logo

华为开发者空间,是为全球开发者打造的专属开发空间,汇聚了华为优质开发资源及工具,致力于让每一位开发者拥有一台云主机,基于华为根生态开发、创新。

更多推荐