Linux底层驱动之树莓派IO口操作

一、Linux中的地址

       Linux内核中,又三种地址:总线地址、物理地址和虚拟地址。
       总线地址: 一种计算机总线,CPU或者有DMA能力的单元中的内存组件或者物理地址在总线上有相对应的地址。比如CPU的寄存器也会在总线上有单独的地址,但是总线地址不等于真实的寄存器。又比如类似于IIC设备在IIC总线上的地址。总线的宽度决定了CPU能够访问内存的范围,32位的最大只能识别4G的内存;
       物理地址: 硬件中的实际地址或绝对地址:比如磁盘地址,比如51的寄存器,就是物理地址;
       虚拟地址: 操作系统运行在保护模式下所使用的地址,也叫逻辑地址,它是基于算法的,处于软件层面。
       树莓派IO口的地址介绍是物理地址,挂载到Linux内核虚拟地址,编程使用的是虚拟地址

二、树莓派IO口的寄存器

       树莓派使用的是(博通)BCM 2835的芯片,芯片提供了54个IO口,对应了树莓派的 BCM ,先介绍几个IO口的寄存器:
在这里插入图片描述

1.GPFSELx IO口模式配置寄存器

在这里插入图片描述
Address :寄存器的基地址,也是寄存器的物理地址;
Field Name:寄存器名;
Descriptin: 寄存器说明;
       GPFSEL0是pin0~pin9的配置寄存器,GPFSEL1是pin10~pin19的配置寄存器,以此类推,GPFSEL5就是pin50~pin53的配置寄存器。每个pin在寄存器上的位置及模式说明:
在这里插入图片描述
       是不是看出了一个规律:管教号 x 3=寄存器的位置。例如pin5,它的位置就是 5x3=15。

2.GPSET0,GPSET1 IO设置寄存器

在这里插入图片描述
GPSET0: pin0~pin31的设置寄存器,1位高电平,0为低电平,复位后为0:
在这里插入图片描述
GPSET1: pin32~pin53的设置寄存器,1位高电平,0为低电平,复位后为0:
在这里插入图片描述

3.GPCLR0,GPCLR1 IO口清除寄存器

在这里插入图片描述
GPCLR0:pin0~pin31的清除寄存器,1位高电平,0为低电平,复位后为0;
在这里插入图片描述
GPCLR1:pin31~pin54的清除寄存器,1位高电平,0为低电平,复位后为0;
在这里插入图片描述
       有这三种寄存器,就操作树莓派的IO口输入高低电平了,但是不要忘了,操作树莓派的IO是操作虚拟地址,而不是上面的 0x7E20…,因此在编程之前一定要查看物理地址的映射 使用:

cat /proc/iomen

在这里插入图片描述
       我的树莓派4B对 0x7E20 0000的映射在虚拟地址 0xFE20 0000上。所以编程时,以上几种寄存器的基地址应该是:0xFE20 000;

三、驱动编程实现

       我们来给树莓派的 GPIO.1(对应 BCM的 pin18 )写驱动,达到操作它输出高低电平,
在这里插入图片描述
驱动程序代码:

#include <linux/fs.h>
#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/init.h>
#include <linux/device.h>
#include <linux/uaccess.h>
#include <asm/io.h>

#define v_uint volatile unsigned int
#define pin 18

static volatile int pin_num=(pin%10)*3;

#define _GPFSEL1 0xFE200004  
#define _GPSET0  0xFE20001C
#define _GPCLR0  0xFE200028

v_uint *GPFSEL1 = NULL;
v_uint *GPSET0  = NULL;
v_uint *GPCLR0  = NULL;

static struct class  *pin18_class;
static struct device *pin18_dev;
static dev_t devno;     //device numble

static int major=232; //major device numble 
static int minor=2;     //minor device numble
static char *module_name = "pin18";//device name

static int pin18_open(struct inode *inode,struct file *file)
{
        printk("%s_open\n",module_name);
		*GPFSEL1 &= ~(0x6 << pin_num);
        *GPFSEL1 |=  (0x1 << pin_num);

        return 0;
};

static ssize_t pin18_write(struct file *file,const char __user *buf,size_t count,loff_t *ppos)
{
        int usercmd;

        printk("%s_write\n",module_name);

        copy_from_user(&usercmd,buf,count);
        printk("get value\n");
        if(usercmd == 1){
                *GPSET0 |= 0x1<< pin;  //设置pin18 口为1
                printk("%s = 1\n",module_name);

        }else if(usercmd ==0){
                *GPCLR0 |= 0x1<< pin; //清除 pin18 口
                printk("%s = 0\n",module_name);
        }else{
                printk("undo\n");
        }

        return 0;
}

static struct file_operations pin18_fops ={
        .owner=THIS_MODULE,
        .open=pin18_open,
        .write=pin18_write,
};

static int __init pin18_dev_init(void)
{
        int ret;
        printk("inmod driver %s success\n",module_name);
        devno =MKDEV(major,minor);
        ret=register_chrdev(major,module_name,&pin18_fops);
		pin18_class=class_create(THIS_MODULE,"mydemo2");
        pin18_dev =device_create(pin18_class,NULL,devno,NULL,module_name);

        GPFSEL1= (v_uint *)ioremap(_GPFSEL1,4);//IO口地址映射
        GPSET0 = (v_uint *)ioremap(_GPSET0,4);
        GPCLR0 = (v_uint *)ioremap(_GPCLR0,4);
        return 0;
}
static void __exit pin18_exit(void)
{
        iounmap(GPFSEL1);
        iounmap(GPSET0);
        iounmap(GPCLR0);

        device_destroy(pin18_class,devno);
        class_destroy(pin18_class);
        unregister_chrdev(major,module_name);
}
module_init(pin18_dev_init);
module_exit(pin18_exit);
MODULE_LICENSE("GPL v2");

切记:一定在初始化的时候,把内存映射到虚拟空间,用 ioremap()函数,而卸载使用 iounmap()函数
       对其进行交叉编译(像看驱动的编译请跳转:驱动的编译)过后,发送到树莓派安装,写一个测试程序,代码如下:

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

int main()
{
        int fd;
        int cmd=1;
        fd=open("/dev/pin18",O_RDWR);
        if(fd<0){
                perror("reson");
                return -1;
        }
        printf("input 1/0:");
        scanf("%d",&cmd);
        fd=write(fd,&cmd,sizeof(int));
        return 0;
}

功能:输入1为GPIO.1输出高电平,输入0为GPIO.1输出低电平,运行结果:
在这里插入图片描述
师承(某(dou)音)上官可编程

  • 3
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 5
    评论
### 回答1: 对于Linux底层驱动面试,以下是我个人的回答: 首先,Linux底层驱动是指与硬件设备交互的一组软件组件。它们允许操作系统与硬件之间的通信和交互。在Linux系统中,驱动程序通常作为内核模块加载到内核中,以支持设备的正常运行。 在面试中,考官可能会关注以下几个方面: 1. 驱动程序的开发 - 面试官可能会问您有关驱动程序开发的经验和技能。他们可能会要求您解释您在开发过程中使用的工具、技术和流程,并问您如何处理常见的驱动程序开发问题。 2. 设备和驱动程序的关系 - 面试官可能会问您对设备和驱动程序之间关系的理解。您需要解释设备如何与驱动程序进行通信以及操作系统如何使用驱动程序来访问设备的功能。 3. 内核模块的加载和卸载 - 面试官可能会问您有关内核模块加载和卸载的知识。您需要解释如何在Linux系统中加载和卸载驱动程序,并讨论在加载和卸载过程中可能遇到的问题和解决方法。 4. 设备树(DTS) - Linux系统中的设备树是描述硬件设备的数据结构。面试官可能会问您对设备树的理解以及如何在驱动程序中使用设备树来配置硬件设备。 5. 调试和故障排除 - 面试官可能会问您关于驱动程序调试和故障排除的问题。您需要解释您在调试和故障排除过程中使用的工具和技术,并描述一些常见的驱动程序问题及其解决方法。 总的来说,面试官希望了解您在Linux底层驱动开发方面的知识和经验。除了理论知识,他们可能还会考察您的实际操作和解决问题的能力。因此,在面试前应该准备并熟悉相关概念和技术,同时也要对常见的驱动程序开发问题进行思考和准备。 ### 回答2: Linux底层驱动面试涉及到Linux内核,对于驱动程序开发有一定的要求。以下是对Linux底层驱动面试的回答: 首先,Linux底层驱动面试通常会要求面试者对Linux内核的工作原理和架构有一定的了解。面试官可能会问及Linux内核的组成、内存管理、进程调度、文件系统等方面的问题,以检验面试者对于Linux内核的掌握程度。 其次,面试者应具备C语言编程技能和驱动开发经验。面试官可能会询问面试者在驱动开发方面的项目经历,对于驱动程序的编写、调试和优化等方面的经验和能力。 另外,面试者还需要了解Linux设备模型和驱动框架。Linux设备模型包括字符设备、块设备、网络设备等,面试者需要了解设备的注册、初始化、IO操作等流程。面试者还需要了解Linux驱动框架,如Platform驱动、PCI驱动、USB驱动等,面试者需要知道如何编写针对特定设备的驱动程序,并能够解释驱动程序的加载、绑定和解绑过程。 最后,面试者应具备问题解决能力和团队合作精神。Linux底层驱动开发涉及到复杂的问题和困难的调试过程,面试者需要展示自己解决问题的能力,并能够与团队合作进行系统的调试与优化。 综上所述,Linux底层驱动面试需要具备对Linux内核的深入理解、熟练的C语言编程和驱动开发经验、对Linux设备模型和驱动框架的熟悉,以及问题解决能力和团队合作精神。希望以上回答对您有所帮助。 ### 回答3: Linux底层驱动面试主要聚焦于以下几个方面: 首先,需要掌握Linux操作系统的基本原理和体系结构。这包括Linux内核的基本组成、内核模块以及驱动的加载和卸载机制等。熟悉Linux系统的启动流程以及内核的初始化过程也是必备的知识。 其次,熟悉Linux设备驱动的框架和模型。Linux的设备驱动模型将驱动程序分为字符设备、块设备和网络设备等类型,并提供了相应的框架和接。面试中需要解释和展示对这些框架和接的理解和使用经验。 接下来,需要了解如何编写和调试Linux驱动程序。这包括如何使用标准的Linux API来开发设备驱动、如何调试和优化驱动程序以及如何处理常见的错误和异常情况等。面试官可能会要求候选人描述自己的开发经验和解决问题的能力。 此外,对于特定设备的驱动开发经验也是面试中的加分项。例如,有经验开发网络驱动、存储设备驱动或声卡驱动等的候选人会更受面试官的关注。在回答问题时,可以结合自身的经验和项目来具体说明自己的能力和技术深度。 最后,沟通能力和团队合作精神也是Linux底层驱动面试中需要考察的因素。Linux驱动开发往往需要与硬件工程师、内核开发者和应用程序开发者进行密切合作。面试官可能会通过场景题或者工作经验来考察候选人的沟通和协作能力。 综上所述,Linux底层驱动面试需要候选人掌握Linux操作系统的基本原理和体系结构、熟悉设备驱动模型和编写调试驱动程序的方法、具备特定设备驱动开发经验、具备良好的沟通和团队合作能力。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值