嵌入式学习——初步驱动led

驱动开发笔记
1;环境搭建
1)ping通
修改网络配置:vi /etc/network/interfaces 修改成static ip为192.168.1.141
修改网络虚拟编辑器:为有线网卡
重启网卡(ifdown eth0 ifup eth0)或者重启ubuntu(shutdown -r now )
2)设置bootcmd使开发板通过tftp下载自己建立的内核源码树编译得到的
set bootcmd ‘tftp 0x30008000 zImage;bootm 0x30008000’(这个是tftp下载的)
bootcmd=movi read kernel 30008000; bootm 30008000(只是烧录到inand中启动的)
我的tftp目录在/tftpboot下,编译好的zImage要复制到这来,每次开发板启动都是动态到这来下载内核

3)将我们配置的rootfs放到我们配置的nfs服务器的目录下
showmount localhost -e查看我们nfs配置的目录

4)make menuconfig 修改配置 支持nfs

5)修改开发板上环境变量
serverip 要保证为我们tftp服务器ip
bootcmd 设置为tftp下载
bootarg 设置为rootfs从nfs下载
setenv bootargs=console=ttySAC2,115200 root=/dev/mmcblk0p2 rw init=/linuxrc rootfstype=ext3

setenv bootargs root=/dev/nfs nfsroot=192.168.1.141:/root/rootfs/rootfs ip=192.168.1.20:192.168.1.141:192.168.1.1:255.255.255.0::eth0:off init=/linuxrc console=ttySAC2,115200

6)修改开发板上命令行密码,将密码去掉
root/rootfs/rootfs/etc/shadow文件

7)rootfs的文件都在开发板下的根目录下 cd /即可,将建立的app可执行程序和.ko模块都放在根文件下
剩下的就是insmod lmod modinfo

8)创建设备文件
mknod /dev/xxx c 主设备号 次设备号

9)应用程序中通过操作设备文件来调用驱动中对应函数

10)驱动文件中操作硬件寄存器应才用虚拟地址
两种方法 静态虚拟映射表和动态虚拟地址映射表,将地址转换一下,操作跟逻辑里面的可以一样

注意事项
1;编译.ko模块的内核要和等下去运行的内核要是同一个

代码

makefile基本是模版不用改

#ubuntu的内核源码树,如果要编译在ubuntu中安装的模块就打开这2个
#KERN_VER = $(shell uname -r)
#KERN_DIR = /lib/modules/$(KERN_VER)/build 


# 开发板的linux内核的源码树目录
KERN_DIR = /root/derive/kernel

#-m表示编译成模块
obj-m   += module_test.o    

#-C表示进入目录  M记录原路径便于返回
all:
    make -C $(KERN_DIR) M=`pwd` modules 
    arm-linux-gcc app.c -o app

cp:
    cp *.ko /root/rootfs/rootfs/driver_test
    cp app /root/rootfs/rootfs/driver_test

.PHONY: clean   
clean:
    make -C $(KERN_DIR) M=`pwd` modules clean
    rm app

总结
模块的makefile非常简单,本身并不能完成模块的编译,而是通过make -C进入到内核源码树下借用内核源码的体系来完成模块的编译链接的。这个Makefile本身是非常模式化的,3和4部分是永远不用动的,只有1和2需要动。1是内核源码树的目录,你必须根据自己的编译环境

/

驱动代码,尽量只包含驱动,操作寄存器代码,逻辑代码尽量放到应用层去

#include        // module_init  module_exit
#include          // __init   __exit
#include            //包含file_operations
#include 
#include 
#include 
#include 
#include 
#define GPJ0CON     S5PV210_GPJ0CON
#define GPJ0DAT     S5PV210_GPJ0DAT

#define rGPJ0CON    *((volatile unsigned int *)GPJ0CON)
#define rGPJ0DAT    *((volatile unsigned int *)GPJ0DAT)

#define GPJ0CON_pa      0xE0200240
unsigned int *rGPJ0CON_pa;

char kernelbuf[100];
static int test_open(struct inode *inode, struct file *file)
{
    printk(KERN_INFO "chrdev_init helloworld init\n");
    rGPJ0CON = 0x11111111;
    //rGPJ0DAT = ((0<<3) | (0<<4) | (0<<5));
    return 0;
}

static int test_release(struct inode *inode, struct file *file)
{
    printk(KERN_INFO "chrdev_exit helloworld exit\n");
    //rGPJ0DAT = ((1<<3) | (1<<4) | (1<<5));
    return 0;
}
ssize_t test_read(struct file *file, char __user *buf, size_t size, loff_t *ppos)
{
    printk(KERN_INFO "test_read\n");
    copy_to_user(buf, kernelbuf, size);
}
//buf是应用层的要需要写入的buf,size需要写入的大小
static ssize_t test_write(struct file *filp, const char __user *buf,size_t size, loff_t *ppos)
{
    printk(KERN_INFO "test_write\n");
    copy_from_user(kernelbuf, buf, size);//这里的大小一般是两个buf中取小的
    if(kernelbuf[0] == '1')
    {
        rGPJ0DAT = ((0<<3) | (0<<4) | (0<<5));
    }
    else if(kernelbuf[0] == '0')
    {
        rGPJ0DAT = ((1<<3) | (1<<4) | (1<<5));
    }
}
//第一步定义file_operations结构体  查内核复制
static const struct file_operations test_fops = {
    .owner      = THIS_MODULE,   
    .open       = test_open,
    .release    = test_release,
    .read       = test_read,
    .write      = test_write,
};
int myret = -1;
//第二步定义注册注销             查内核复制



// 模块安装函数
static int __init chrdev_init(void)
{   

    myret = register_chrdev(0, "test_modu", &test_fops);   //margin传0表示由内核自动分配
    if(myret < 0)
    {
        printk(KERN_ERR "chrdev_init helloworld init\n");  //关于打印级别或者错误编号意义
        return  -EINVAL; //errno-base.h中定义了错误的编号    如果不知道可以去搜一个然后追
                                                        //定义所有的头文件
    }
    printk(KERN_INFO "open major %d\n", myret);
    //test_open();
    //动态映射  
    if (!request_mem_region(GPJ0CON_pa, 8, "GPJ0_LED")) 
    {
        printk(KERN_ERR "request_mem_region error\n");
        return -EINVAL;
    }
    printk(KERN_INFO "request_mem_region success\n");
    rGPJ0CON_pa = ioremap(GPJ0CON_pa, 8);

    *rGPJ0CON_pa = 0x11111111;
    *(rGPJ0CON_pa + 1) = ((0<<3) | (0<<4) | (0<<5));

    return 0;
}

// 模块卸载函数
static void __exit chrdev_exit(void)
{
    unregister_chrdev(myret, "test_modu");
    //test_release();
    *(rGPJ0CON_pa + 1) = ((1<<3) | (1<<4) | (1<<5));
    iounmap(rGPJ0CON_pa);
    release_mem_region(GPJ0CON_pa, 8);
}


module_init(chrdev_init);
module_exit(chrdev_exit);

// MODULE_xxx这种宏作用是用来添加模块描述信息
MODULE_LICENSE("GPL");              // 描述模块的许可证
MODULE_AUTHOR("aston");             // 描述模块的作者
MODULE_DESCRIPTION("module test");  // 描述模块的介绍信息
MODULE_ALIAS("alias xxx");          // 描述模块的别名信息








应用层

#include 
#include 
#include 
#include 
#include 

#define Myname "/dev/test"  //mknod的那个对应的设备文件
char appbuf[100];
int main()
{
    int ret = -1;
    int i = 0;
    ret = open(Myname, O_RDWR);

    if(ret < 0)
    {
        printf("open %s error\n", Myname);
        return -1;
    }
    printf("open %s success\n", Myname);

    //读写文件间接操作驱动的读写函数
    //write(ret, "hellowhold", 10);
    //(ret, appbuf, 10);
    //sleep(10);

    printf("下面驱动led灯  输入on  or off or flash\n");
    while(1)
    {
        scanf("%s",appbuf);
        if(!strcmp(appbuf, "on"))
        {
            write(ret, "1", 1);
        }
        else if(!strcmp(appbuf, "off"))
        {
            write(ret, "0", 1);
        }
        else if(!strcmp(appbuf, "flash"))
        {
            for(i = 0; i < 3; i++)
            {
                write(ret, "1", 1);
                sleep(1);
                write(ret, "0", 1);
                sleep(1);
            }
        }
    }

    //关闭
    close(ret);
    printf("read appbuf = %s\n", appbuf);
    return 0;
}

总结:
1;完成MODULE_xxx的一些宏(描述模块的一下基本信息
2;module_init(chrdev_init);module_exit(chrdev_exit);绑定模块建立和撤销时对应insmod和rmmod调用的对应函数
3;注册驱动调用register_chrdev,
关键参数
主设备号若传0则表示内核自动分配有返回值返回,
file_operations结构体指针(里面只要是一些函数指针)用来挂接驱动和内核api的,就是应用层调用这个函数则会调用该结构体指向的对应函数,从内核复制并修改,并且完成内核函数指针指向的函数进行定义。
4;创建应用程序,主要是io操作,操作的就是mknod建立的设备文件,这样才会完成在应用层对该文件操作,在驱动层则可以接收,如应用层open设备文件,则在驱动层相当于调用file_operations结构体中open元素指向的函数。
5;在驱动层操作寄存器的虚拟地址来进行读写操作来操作硬件

采用驱动新接口,注册注销时都分两步,设备号注册和驱动注册两步
代码,只需要修改init绑定的函数

// 模块安装函数
static int __init chrdev_init(void)
{   
/*
    旧接口的注册
    myret = register_chrdev(0, "test_modu", &test_fops);   //margin传0表示由内核自动分配
    if(myret < 0)
    {
        printk(KERN_ERR "chrdev_init helloworld init\n");  //关于打印级别或者错误编号意义
        return  -EINVAL; //errno-base.h中定义了错误的编号    如果不知道可以去搜一个然后追
                                                        //定义所有的头文件
    }
    printk(KERN_INFO "open major %d\n", myret);
*/
    //新接口注册
    //第一步注册设备号   先MKDEV得到设备号,再利用register_chrdev_region来注册
    int retval;
    //mydev = MKDEV(200,0);//将主设备号和次设备号的起始序列合并成设备号
    //retval = register_chrdev_region(mydev, MYCOUNT, "test_modu");
    //分配设备号
    retval = alloc_chrdev_region(&mydev, 0, MYCOUNT, "test_modu");
    if (retval < 0) {
        printk(KERN_ERR "Unable to register minors for test_modu\n");
        goto flag1;
    }
    printk(KERN_INFO "register_chrdev_region success\n");
    printk(KERN_INFO "major %d  minors %d\n", MAJOR(mydev), MINOR(mydev));


    //绑定注册驱动
    pmycdev = cdev_alloc();  //相当对内核里面申请堆内存
    if (!pmycdev)
    {
        printk(KERN_ERR "cdev_alloc error\n");
        goto flag2;
    }

    cdev_init(pmycdev, &test_fops);//绑定初始化mycdev
    retval = cdev_add(pmycdev, mydev, MYCOUNT);//注册将file_operation结构体与设备号绑定
    if (retval) {
        printk(KERN_ERR "Unable to get testmodu major \n");
        goto flag3;
    }


    //test_open();
    //动态映射  
    if (!request_mem_region(GPJ0CON_pa, 8, "GPJ0_LED")) 
    {
        printk(KERN_ERR "request_mem_region error\n");
        goto flag4;
    }
    printk(KERN_INFO "request_mem_region success\n");
    rGPJ0CON_pa = ioremap(GPJ0CON_pa, 8);

    *rGPJ0CON_pa = 0x11111111;
    *(rGPJ0CON_pa + 1) = ((0<<3) | (0<<4) | (0<<5));

    return 0;

//一个中途出错的倒影式错误处理方法
flag4:
    cdev_del(pmycdev);
flag3:
    cdev_del(pmycdev);
flag2:
    unregister_chrdev_region(mydev, MYCOUNT);
flag1:
    return -EINVAL;

}

// 模块卸载函数
static void __exit chrdev_exit(void)
{
/*
旧的注销方法
    unregister_chrdev(myret, "test_modu");
    */
    //驱动注销
    cdev_del(pmycdev);
    //设备号注销
    unregister_chrdev_region(mydev, MYCOUNT);

    //test_release();
    *(rGPJ0CON_pa + 1) = ((1<<3) | (1<<4) | (1<<5));
    iounmap(rGPJ0CON_pa);
    release_mem_region(GPJ0CON_pa, 8);
}

你可能感兴趣的:(嵌入式学习)