1.Linux如何找到设备文件

通过 文件名和设备号

设备号 : 主设备号和次设备号

主设备号区分不同种类的设备

次设备号区分同一类型不同的设备

Linux的驱动链表会管理这些设备驱动

1.添加(编写完驱动程序加载到内核)

2.查找 (调用驱动程序,用户层去调用open)

驱动插入链表的顺序由设备号检索


2.驱动开发

驱动的开发就是 添加驱动和调用驱动

添加驱动: 设备名  设别号  驱动函数(操作寄存器驱动IO口)

用户态open()会进入内核态  此时会发生软中断 中断号0x80

 image.png


3.驱动框架代码

#include <linux/fs.h> //file_operations声明
#include <linux/module.h>    //module_init  module_exit声明
#include <linux/init.h>      //__init  __exit 宏定义声明
#include <linux/device.h> //class  devise声明
#include <linux/uaccess.h>   //copy_from_user 的头文件
#include <linux/types.h>     //设备号  dev_t 类型声明
#include <asm/io.h>          //ioremap iounmap的头文件


static struct class *pin4_class;  
static struct device *pin4_class_dev;

static dev_t devno;                //设备号
static int major =231;     //主设备号
static int minor =0;    //次设备号
static char *module_name="pin4";   //模块名




//led_open函数
static int pin4_open(struct inode *inode,struct file *file)
{
    printk("pin4_open\n");  //内核的打印函数和printf类似
   

    
    return 0;
}

//led_write函数
static ssize_t pin4_write(struct file *file,const char __user *buf,size_t count, loff_t *ppos)
{


    return 0;
}

static struct file_operations pin4_fops = {

    .owner = THIS_MODULE,
    .open  = pin4_open,
    .write = pin4_write,
};

int __init pin4_drv_init(void)   //程序的真实入口
{

    int ret;
    devno = MKDEV(major,minor);  //创建设备号
    ret   = register_chrdev(major, module_name,&pin4_fops);  //注册驱动  告诉内核,把这个驱动加入到内核驱动的链表中

    pin4_class=class_create(THIS_MODULE,"myfirstdemo");
    pin4_class_dev =device_create(pin4_class,NULL,devno,NULL,module_name);  //创建设备文件

 
    return 0;
}

void __exit pin4_drv_exit(void)
{

    device_destroy(pin4_class,devno);//销毁设备
    class_destroy(pin4_class);//销毁类
    unregister_chrdev(major, module_name);  //卸载驱动

}

module_init(pin4_drv_init);  //入口
module_exit(pin4_drv_exit);
MODULE_LICENSE("GPL v2");

4.驱动测试

(1)把驱动文件放到上位机的driver/目录下

(2)修改Makefile  把其编译为模块

把 obj-m   += xxx. o 加入到makefile中

(3)编译成模块

ARCH=arm CROSS_COMPILE=交叉编译工具 KERNEL=内核版本  make modules

(4) 装载驱动到驱动链表

sudo   insmod xxx.ko

  ( 5)装载驱动会自动生成设备  比如 /dev/pin4,通过chmod 666 /dev/pin4 给其执行权限

  (6)运行测试程序调用驱动

 

内核驱动装载: sudo  insmod xxx.ko

内核驱动卸载  :sudo  rmmod xxx 不用写ko

查看内核模块:lsmod

内核的printk相当于用户态的printf  通过dmesg查看内核打印信息

 

5,地址

1.总线地址

总线地址就是cpu可以访问内存的地址范围

装了32位系统的计算机最大可以访问2 ^32bit

Linux可以通过cat /proc/meminfo

2.物理地址

硬件的实际地址或绝对地址就是硬件的物理地址

3.虚拟地址

虚拟地址就是逻辑地址,是基于算法的地址,软件层面的地址。

 image.png

当物理地址不够用时,会通过MMU单元映射为虚拟地址,通过虚拟地址来保证程序的运行。上层应用操作的地址都是虚拟地址。

页表是内存非连续分区分配的基础,实现从逻辑地址转化成物理地址。页表是mmu管理的


6.树莓派驱动开发

//pin4_driver2.c
#include <linux/fs.h> //file_operations声明
#include <linux/module.h>    //module_init  module_exit声明
#include <linux/init.h>      //__init  __exit 宏定义声明
#include <linux/device.h> //class  devise声明
#include <linux/uaccess.h>   //copy_from_user 的头文件
#include <linux/types.h>     //设备号  dev_t 类型声明
#include <asm/io.h>          //ioremap iounmap的头文件


static struct class *pin4_class;  
static struct device *pin4_class_dev;

static dev_t devno;                //设备号
static int major =231;     //主设备号
static int minor =0;    //次设备号
static char *module_name="pin4";   //模块名

volatile unsigned int* GPFSL0=NULL;//功能选择 输入或输出
volatile unsigned int* GPSET0=NULL//输出高电平
volatile unsigned int* GPCLR0=NULL//清0
//volatile  的作用
//1.避免编译优化 ,避免系统把地址优化为别的地址
//2.可以直接读出它的值
//IO口起始空间 0x3f000000
//led_open函数
static int pin4_open(struct inode *inode,struct file *file)
{
    printk("pin4_open\n");  //内核的打印函数和printf类似
    //配置pin4引脚为输出引脚   bit14-12配置001
    *GPFSL0 &=~(0x6<<12);//0x6为110 让第14 13位为0   给110左移12位11正好对着14 13位  在取反14 13位就是0了(计算机的位是从0开始的)
    //接下来让第12为1
    *GFPSL0 |=(0x1<<12);//这样1正好在12位   与它或运算第12位恰好为1
    
    return 0;
}

//led_write函数
static ssize_t pin4_write(struct file *file,const char __user *buf,size_t count, loff_t *ppos)
{
int userCmd;
copy_from_user(&userCmd,buf,count);//读取用户态应用的输出并用userCmd去接收
//读取上层应用write的值来判断输出1还是0
printk("get value\n");
if(userCmd==1){
//拉高电平
printk("set 1\n");
*GPSET0 |= 0x1 << 4;//让pin4引脚为1
}
else if(userCmd==0){
//拉低电平
printk("set 0\n");
*GPCLR0 |= 0x1 << 4;//让pin4引脚为0
}else{
printk("undo\n");
}
    return 0;
}

static struct file_operations pin4_fops = {

    .owner = THIS_MODULE,
    .open  = pin4_open,
    .write = pin4_write,
};

int __init pin4_drv_init(void)   //程序的真实入口
{

    int ret;
    devno = MKDEV(major,minor);  //创建设备号
    ret   = register_chrdev(major, module_name,&pin4_fops);  //注册驱动  告诉内核,把这个驱动加入到内核驱动的链表中

    pin4_class=class_create(THIS_MODULE,"myfirstdemo");
    pin4_class_dev =device_create(pin4_class,NULL,devno,NULL,module_name);  //创建设备文件
 
GPFSEL0 = (volatile unsigned int *)ioremap(0x3f200000,4);
//0x3f200000是物理地址  根据页表 1个字节按照虚拟地址是4个字节  ioremap把物理地址转化为虚拟地址
GPSET0  = (volatile unsigned int *)ioremap(0x3f20001C,4);
GPCLR0  = (volatile unsigned int *)ioremap(0x3f200028,4);

    return 0;
}

void __exit pin4_drv_exit(void)
{
iounmap(GPFSEL0);//它的作用就是把映射过来的物理地址还原,释放掉虚拟地址
iounmap(GPSET0);
iounmap(GPCLR0);

    device_destroy(pin4_class,devno);//销毁设备
    class_destroy(pin4_class);//销毁类
    unregister_chrdev(major, module_name);  //卸载驱动

}

module_init(pin4_drv_init);  //入口
module_exit(pin4_drv_exit);
MODULE_LICENSE("GPL v2");

 

7.用户态测试代码

//test.c
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdio.h>

int main()
{
        int fd;
        int cmd;

        fd = open("/dev/pin4",O_RDWR);
        if(fd < 0){
                printf("open failed\n");
                perror("reson");
        }else{
                printf("open success\n");
        }

        printf("请输入0 / 1\n 0:设置pin4为低电平\n 1:设置pin4为高电平\n");
        scanf("%d",&cmd);

        if(cmd == 0){
                printf("pin4设置成低电平\n");
        }else if(cmd == 1){
                printf("pin4设置成高电平\n");
        }

        fd = write(fd,&cmd,4);//写一个数字1,写4个字节
        return 0;
}

8.编译

1.把驱动代码拷贝到/driver/char目录下

2.修改Makefile

3.回到源码目录/linux-rpi-4.14.y再执行下面指令编译为模块 生成xxx.ko


ARCH=arm CROSS_COMPILE=arm-linux-gnueabihf- KERNEL=kernel7 make modules

4.把生成的模块拷贝到开发板 scp指令

5.交叉编译测试代码


arm-linux-gnueabihf-gcc test.c -o pin4test

6.把test发送到开发板 scp指令


9.装载驱动模块

1.sudo insmod pin4_drive2.ko装载驱动


如果要卸载的话就用 sudo rmmod指令

2.sudo chmod 666 /dev/pin4 给所有用户执行权限

3.运行 ./pin4test

————————————————

版权声明:本文为CSDN博主「C有点难。」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。

原文链接:https://blog.csdn.net/weixin_45993872/article/details/129479676