与驱动有关的面试问题_驱动工程师面试题

img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上物联网嵌入式知识点,真正体系化!

由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、电子书籍、讲解视频,并且后续会持续更新

需要这些体系化资料的朋友,可以加我V获取:vip1024c (备注嵌入式)

如果你需要这些资料,可以戳这里获取

(1)内存划分

以32位机器为例,cpu最大寻址范围为4G,Linux系统将4G虚拟地址空间划分为高1G,低3G。

低3G虚拟空间属于用户空间,都是经过映射的线性地址,供用户进程空间使用。

高1G并非都是像用户空间一样都是映射过的线性空间,Linux系统将高1G划分为三部分:DMA区,常规区,高端内存,其中0-896都是映射过的线性空间,剩下的896-1024即高端内存,这段高端内存都是未经过映射的虚拟地址,Linux系统利用这些有限的虚拟地址,临时动态的映射到大于896M的物理空间地址,实现了利用有限的虚拟地址访问到物理内存的所有地址。

(2)内存申请函数的对比

malloc用于用户空间进程申请内存空间,kmallc和vmalloc在内核空间使用。

kmalloc申请到的内存空间,是线性连续的,可以用于dma。

vmalloc申请的内存是逻辑连续的,但是物理地址不连续,常用与申请大的内存,请注意vmalloc可能会睡眠,在中断、阻塞的环境下不能使用。

(3)内存映射方式

虚拟地址到物理地址的转化,用户空间和内和空间采用不同的映射机制。用户空间的地址映射经过mmu(内存管理单元)管理。而内核空间的虚拟地址到物理地址的映射是一一对应的,例如虚拟空间地址0xc0000004,对应的物理地址空间地址为:0xc0000004 - 0xc0000000 = 0x04,以此类推。(待求证)。

4、inux中中断的实现机制,tasklet与workqueue的区别及底层实现区别?为什么要区分上半部和下半部?

中断的上下半部_天糊土的博客-CSDN博客_中断下半部

(1)Linux中断分为硬件中断和内部中断(异常),调用过程:外部中断产生->发送中断信号到中断控制器->通知处理器产生中断的中断号,让其进一步处理。即处理器收到来自中断控制器的中断处理请求,保存中断上下文,跳转到中断对应的处理处,(快速完成中断中断上半部,中断上半部返回后执行中断下半部),中断处理函数返回时恢复现场。

(2)tasklet和workqueue,两者都是中断下半部的一种实现方法。区别在于tasklet属于中断上下文,支持smp、不可睡眠和阻塞;workqueue基于线程的封装,属于进程上下文,因此支持睡眠、阻塞。

(3)为了能够在中断处理过程中被新的中断打断,将中断处理程序一分为二,上半部登记新的中断,快速处理简单的任务,剩余复杂耗时的处理留给下半部处理。下半部处理过程中可以被中断,上半部处理时不可被中断。

5、Linux的同步机制

常见的同步接口,包括进程同步,信号量,自旋锁,互斥锁,条件变量,读写锁。

多进程并发一般考虑使用信号量机制,在线程并发时多采用互斥锁,条件变量。

(1)条件变量在某些角度就是线程版的信号量实现,因为两者都是在考虑持有锁时间较长情况下使用。

(2)互斥锁,自旋锁一般都是用在持有锁时间不会很长的情况下,在自旋锁有使用意义的前提下,如果持锁时间会非常短则自旋锁效率高于互斥锁(否则应该使用互斥锁,因为互斥锁会持续占有cpu资源,不宜过长,而互斥锁会导致抢不到锁的线程睡眠,进入等待队列)。

(3)互斥锁和自旋锁都可以用在进程上下文,而在中断上下文只能使用自旋锁,因为互斥锁会睡眠。

(4)使用自旋锁的进程不能睡眠,使用信号量的进程可以睡眠。

7、/dev/下面的设备文件是怎么创建出来的?

有三种方式:devfs机制、udev机制、手动创建设备节点。

谈谈个人见解:

devfs机制,从来没用过,应该是2.6以前的内核使用的;

udev机制,其实就是现在常用的device_create()、class_create()这一套接口,所谓udev是上层用户空间程序,是基于驱动中创建使用了这两个接口而起作用的,但是udev在日常开发中几乎接触不到,我们只需在驱动中调用创建节点的这两个API就ok了,剩下的工作就交给udev去做。

mknod ,新手最常用的一种创建设备节点方法,但并非入门后就再没有用途。在某些情境下,或许有人不想使用udev机制,于是把节点创建工作写在脚本里,这样也是无可厚非的。

8、原子操作该怎么理解?

原子操作,就是开始执行到执行结束期间不会被打断的操作单元。

9、insmod一个驱动模块,会执行模块中的哪个函数?rmmod呢?这两个函数在设计上要注意哪些?遇到过卸载驱动出现异常没?是什么问题引起的?

分别会执行module_init()、module_exit()指定的init函数和exit函数。

要注意的就是尽量使在init函数中出现的资源申请及使用有对应的释放操作在exit中,init申请,eixt释放。

卸载出现的异常?那很稀松平常了,大多数都是资源使用完没释放,但是模块却卸载了。

10、在驱动调试过程中遇到过oops没?你是怎么处理的?

遇到过,这种类似的段错误其实最好处理,因为它有call trace,根据堆栈信息去代码里面查看就行了。

如果代码中看不到明显低级错误,就需要printk联机调试,然而这种很少。

11、ioctl和unlock_ioctl有什么区别?

ioctl是老的内核版本中的驱动API,unlock_ioctl是当下常用的驱动API。

区别在于ioctl调用前后,使用了大内核锁,而unlock_ioctl顾名思义就是没加大内核锁的新接口,改变的只是驱动调的方法,用户应用程序调用的接口不变。

大内核锁是Linux hacker在应付多处理器初期提出的一种锁,目的在于当一个处理核心在运行内核时,加上大内核锁,不让其他cpu核心同时运行内核程序,显然这样是有用的,然而这样大大降低了多处理器的存在意义,于是跟随时代更迭,大内核锁被一步一步的剔除,ioctl接口的升级就是典范!我觉得这样的问题太没有意义了,对开发有多大用处?

12、驱动中操作物理绝对地址为什么要先ioremap?

ioremp是内核中用来将外设寄存器物理地址映射到主存上去的接口,即将io地址空间映射到虚拟地址空间上去,便于操作。为什么非要映射呢,因为保护模式下的cpu只认虚拟地址,不认物理地址,给它物理地址它并不帮你做事,所以你要操作外设上的寄存器必须先映射到虚拟内存空间,拿着虚拟地址去跟cpu对接,从而操作寄存器。

11、设备驱动模型三个重要成员是?platform总线的匹配规则是?在具体应用上要不要先注册驱动再注册设备?有先后顺序没?

(1)总线,设备,驱动。

(2)匹配规则:当有一个新的设备挂起时,总线被唤醒,match函数被调用,用device名字去跟本总线下的所有驱动名字去比较,相反就是用驱动的名字去device链表中和所有device的名字比较。如果匹配上,才会调用驱动中的probe函数,否则不调用。

(3)至于先后顺序,鉴于个人理解,不会有影响,不管谁先谁后,bus都会完成匹配工作。

(4)谈谈对Linux设备驱动模型的认识:设备驱动模型的出现主要有三个好处

设备与驱动分离,驱动可移植性增强;

设备驱动抽象结构以总线结构表示,看起来更加清晰明,谁是属于哪一条bus的;

最后,设备与驱动分离,很好的奠定了热插拔机制。

12、linux中RCU原理?

rcu是2.6出现的一种读写锁,可以说是老的读写锁的升级版,主要用在链表这种数据结构上,经典使用场景是多读者少写者的情况,rcu允许多个读者一个写者共同操作数据而不必加锁,这是经典用法,若出现多个写者时,写者与写者之间就得自己手动同步。当要删除一个节点时,删除后并不会马上释放节点,而是会等待在删除动作之前已经开始读该节点的读者都完成读操作之后才会释放此节点,这段时间被称为宽限期。

13、谈谈Linux软中断?

Linux系统中的软中断,是专为一些不是特别要紧的耗时任务而产生的一种机制,多数用在中断处理过程中,典型应用就是用于中断下半部,tasklet机制就是基于软中断的典型下半部应用。

软中断就是结合任务调度、延迟处理等让守护进程去处理一些不是特别紧急又耗时的任务

14、linux系统实现原子操作有哪些方法?

答:提到原子操作,我首先想到的是针对整型的原子操作,atomic_t类型,这里面有一整套针对整型的原子操作API可以调用。既然整型能原子操作,那其他也应该可以吧,结合原子操作的定义,要想对其他类型结构实现原子操作,那就加锁咯,将需要原子操作的部分放在临界区。

15、linux中系统调用过程?

系统调用,比如open()函数,它并不是真正的系统调用实现函数,其实它只是一个c库函数。

内部实现做了两件事,先把系统调用号传递给内核,最后拉起一次软中断,自此cpu进入内核态运行。

内核在软中断向量表中找出对应的中断类型,根据中断类型找到对应的软中断执行函数,然后执行函数根据系统调用号,在系统调用号表里面找到对应的系统调用函数。

16、谈谈linux内核的启动过程(源代码级)?

首先,内核镜像自解压,解压完之后从head.s开始运行,即引导内核,在内核引导期间将会设置内核参数。

随后,跳转到第一个c函数start_kernel(),进入内核启动阶段,在内核启动过程中进行一些必要的硬件初始化工作。

在内核启动最后,挂载文件系统,然后创建第一个用户空间进程,init进程,进一步完成驱动挂载,用户服务初始化工作。

17、谈谈Linux调度原理?

Linux将进程按权限分为两大类,常规进程和实时进程

常规进程对应一种调度算法,实时进程有两种对应着两种不同的调度算法。

进程按照状态又可以分为几种,常见的状态有:运行态、可中断睡眠态、不可中断睡眠态、停止态。处于运行态的进程根据调度算法接受调度在cpu上运行。

18、谈谈对Linux网络子系统的认识?

网络子系统可以概括描述为

应用程序—》系统调用接口(主要是指socket接口)—》协议无关接口(由socket实现,提供一套通用接口支持不同的协议)—》网络协议(包括tcp、udp在内的网络协议)—》设备无关接口(由net_device接口组织的一组通用接口,将网络协议与各种网络设备联系起来)—》设备驱动(即各种网络设备的驱动程序,负责管理具体的网络设备)—》网络设备(具体的网络硬件设备)。

19、内核中申请内存有哪几个函数?有什么区别?

常见的三个接口,kmalloc(),vmalloc(),__get_free_pages()。

kmalloc()操作的空间位于直接映射区(即4G空间中的896M区域),申请到空间物理地址多为连续地址,常用于操作频繁的数据结构,连续地址利于提高访问效率。

对于一些操作不频繁的数据结构可以用vmalloc()申请内存,vmalloc()操作的空间优先选择高端内存,这里申请出的内存物理地址往往不是连续的,所以访问效率不会很高。

__get_free_pages()操作的区域跟kmalloc()相同,位于直接映射区,不同的是它申请的是物理页的整倍数大小的内存。

20、谈谈内核函数mmap的实现机制?

(1)mmap函数,把一个文件映射到一个内存区域,从而我们可以像读写内存一样读写文件,其比单纯调用read/write要快上许多。在某些时候我们可以把内存的内容拷贝到一个文件中实现内存备份,当然也可以把文件的内容映射到内存来恢复某些服务。

(2)mmap实现共享内存也是其主要应用之一,mmap系统调用使得进程之间通过映射同一个普通文件实现共享内存。

21、中断上半部、下半部的实现理解

(1)当中断到来时,要完成的工作往往并不会是短小的,它可能要进行较大量的耗时处理。因此中断处理程序中,不要求立即完成的,可以交由中断下半部完成。

中断上半部主要完成尽可能少的、比较紧急的功能,例如简单地读取寄存器中的中断状态并清除中断标志后就进行“登记中断”的工作。顶半部执行的速度就会很快,可以服务更多的中断请求。

复杂的内容则交由中断下半部来执行,而且中断下半部可以被新的中断打断,这也是下半部和上半部的最大不同。

当然, 如果中断比较简单,就不用区分上下半部。

(2)Linux 系统实现下半部的机制主要有三种:tasklet,workqueue,软中断。

22、tasklet与workqueue的区别及底层实现区别

(1)softirq和tasklet都属于软中断,tasklet是softirq的特殊实现;workqueue是普通的工作队列。

(2)tasklet和workqueue,两者都是中断下半部的一种实现方法。区别在于tasklet属于中断上下文,支持smp、不可睡眠和阻塞;workqueue基于线程的封装,属于进程上下文,因此支持睡眠、阻塞。

23、软中断的理解(与硬件中断的对比)

软中断过程:

(1)当某一软中断时间发生后,首先需要设置对应的中断标记位,触发中断事务(raise_softirq()设置软中断状态bitmap,触发软中断事务)。

(2)然后唤醒守护线程去检测中断状态寄存器(在Linux中 软中断daemon线程函数为do_softirq())。

(3)如果通过查询发现某一软中断事务发生之后,那么通过软中断向量表调用软中断服务程序action()。

和硬件中断的对比:

(1)一个软中断不会去抢占另一个软中断,只有硬件中断才可以抢占软中断,所以软中断能够保证对时间的严格要求。

(2)软中断与硬件中断不同在于,从中断标记到中断服务程序的映射过程。

硬件中断发生之后,CPU需要将硬件中断请求通过向量表映射成具体的服务程序,这个过程是硬件自 动完成的。

但是软中断不是,其需要守护线程去实现这一过程,这也就是软件模拟的中断,故称之为软中断。

24、请简述自旋锁、信号量两个概念,及它们的区别。

(1)自旋锁

自旋锁是专门为防止多处理器并发而引入的一种锁,在内核中大量应用于中断处理等部分。(对于单处理器来说,防止中断处理中的并发可简单采用关闭中断的方式,不需要自旋锁)。

自旋锁最多只能被一个内核任务持有,如果一个内核任务试图请求一个已被争用(已经被持有)的自旋锁,那么这个任务就会一直进行忙循环——旋转——等待锁重新可用。要是锁未被争用,请求它的内核任务便能立刻得到它并且继续进行。

自旋锁可以在任何时刻防止多于一个的内核任务同时进入临界区,因此这种锁可有效地避免多处理器上并发运行的内核任务竞争共享资源。

自旋锁的初衷,是在短期间内进行轻量级的锁定。一个被争用的自旋锁使得请求它的线程在等待锁重新可用的期间进行自旋(特别浪费处理器时间),所以自旋锁不应该被持有时间过长。如果需要长时间锁定的话, 最好使用信号量。

因为自旋锁在同一时刻只能被最多一个内核任务持有,所以一个时刻只有一个线程允许存在于临界区中。这点很好地满足了对称多处理机器需要的锁定服务。在单处理器上,自旋锁仅仅当作一个设置内核抢占的开关。如果内核抢占也不存在,那么自旋锁会在编译时被完全剔除出内核。

简单的说,自旋锁在内核中主要用来防止多处理器中并发访问临界区,防止内核抢占造成的竞争。另外自旋锁不允许任务睡眠(持有自旋锁的任务睡眠会造成自死锁——因为睡眠有可能造成持有锁的内核任务被重新调度,而再次申请自己已持有的锁),它能够在中断上下文中使用。

**死锁:**假设有一个或多个内核任务和一个或多个资源,每个内核都在等待其中的一个资源,但所有的资源都已经被占用了。这便会发生所有内核任务都在相互等待,但它们永远不会释放已经占有的资源,于是任何内核任务都无法获得所需要的资源,无法继续运行,这便意味着死锁发生了。自死琐是说自己占有了某个资源,然后自己又申请自己已占有的资源,显然不可能再获得该资源,因此就自缚手脚了。

自旋锁的基本形式如下:
spin_lock(&mr_lock);
//临界区
spin_unlock(&mr_lock);

(2)信号量

Linux中的信号量是一种睡眠锁。如果有一个任务试图获得一个已被持有的信号量时,信号量会将其推入等待队列,然后让其睡眠。这时处理器获得自由去执行其它代码。当持有信号量的进程将信号量释放后,在等待队列中的一个任务将被唤醒,从而便可以获得这个信号量。

信号量的睡眠特性,使得信号量适用于锁会被长时间持有的情况;只能在进程上下文中使用,因为中断上下文中是不能被调度的;另外当代码持有信号量时,不可以再持有自旋锁。

信号量基本使用形式为:
static DECLARE_MUTEX(mr_sem);//声明互斥信号量
if(down_interruptible(&mr_sem))
//可被中断的睡眠,当信号来到,睡眠的任务被唤醒
//临界区
up(&mr_sem);

(3)信号量和自旋锁区别

1、如果代码需要睡眠(往往发生在和用户空间同步时),使用信号量是唯一的选择。由于不受睡眠的限制,使用信号量通常来说更加简单一些。

2、如果需要在自旋锁和信号量中作选择,应该取决于锁被持有的时间长短。理想情况是所有的锁都应该尽可能短的被持有,但是如果锁的持有时间较长的话,使用信号量是更好的选择。

3、信号量不同于自旋锁,它不会关闭内核抢占,所以持有信号量的代码可以被抢占。这意味者信号量不会对影响调度反应时间带来负面影响。

26、什么是GPIO?

general purpose input/output ,即可以把这些引脚拿来用作任何一般用途的输入输出,例如用一根引脚连到led的一极来控制它的亮灭,也可以用一根(一些)引脚连到一个传感器上以获得该传感器的状态,这给cpu提供了一个方便的控制周边设备的途经。

27、在Linux C中,ls这个命令是怎么被执行的?

使用fork创建一个进程或exec函数族覆盖原进程。

28、写一段 C 代码让程序跳转到地址是 0x8000 0000 的位置执行

(*(void(*)(void))0x100000)();或者((void(*)(void))0x100000)();

29、简要叙述进程和线程这两个概念。

进程是指一个程序在一个数据集合上的一次运行过程。

线程是进程中的一个实体,是被系统独立调度和执行的基本单位。

进程是程序的一次执行。线程可以理解为进程中执行的一段程序片段。

30、在一个只有128M内存并且没有交换分区的机器上,说说下面两个程序的运行结果

1、
#define MEMSIZE 1024*1024
int count = 0;
void *p = NULL;
while(1) {
  p = (void *)malloc(MEMSIZE);
  if (!p) break;
  printf(“Current allocation %d MB\n”, ++count);
}
2、
while(1) {
  p = (void *)malloc(MEMSIZE);
  if (!p) break;
  memset(p, 1, MEMSIZE);
  printf(“Current allocation %d MB\n”, ++count);
}

第一道程序分配内存但没有填充,编译器可能会把内存分配优化掉,程序死循环。

第二道,程序分配内存并进行填充,系统会一直分配内存,直到内存不足,退出循环。

31、请定义一个宏,比较两个数a、b的大小,不能使用大于、小于、if语句

如果可以用小于号的话,可以这么写:#define compare(a,b) ((a-b)<0 ? -1 : ((a-b) == 0 ? 0 : 1))
这样一来的话:
compare(a,b) == -1表示a<b
compare(a,b) == 0表示a==b
compare(a,b) == 1表示a>b

但是如今不能用小于号,那么我们怎样不用小于号来判断一个数字是否小于0呢?我们可以用:abs(t) != t ? 1 : -1来表示。
也即如果abs(t) != t那么t<0,否则t>=0。也即t<0等价于(abs(t) != t ? 1 : -1) == 1那么前面的宏写成如下:
#define compare(a,b) ((abs(a-b) != (a-b) ? 1 : -1) == 1 ? -1 : ((a-b) == 0 ? 0 : 1))
代码:

//利用宏比较大小
#include 
#include 

#define compare(a,b) ((abs(a-b) != (a-b) ? 1 : -1) == 1 ? -1 : ((a-b) == 0 ? 0 : 1))

int main()
{
    int a1 = -1, b1 = 2, a2 = 3, b2 = 3, a3 = 4, b3= 2;

cout << compare(a1, b1) << endl << compare(a2, b2) << endl << compare(a3, b3) << endl;
    
    return 0;
}

收集整理了一份《2024年最新物联网嵌入式全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升的朋友。
img
img

如果你需要这些资料,可以戳这里获取

需要这些体系化资料的朋友,可以加我V获取:vip1024c (备注嵌入式)

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人

都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

? 0 : 1))

int main()
{
    int a1 = -1, b1 = 2, a2 = 3, b2 = 3, a3 = 4, b3= 2;

cout << compare(a1, b1) << endl << compare(a2, b2) << endl << compare(a3, b3) << endl;
    
    return 0;
}

收集整理了一份《2024年最新物联网嵌入式全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升的朋友。
[外链图片转存中…(img-hROpwDdS-1715799407765)]
[外链图片转存中…(img-iDfkEIzP-1715799407765)]

如果你需要这些资料,可以戳这里获取

需要这些体系化资料的朋友,可以加我V获取:vip1024c (备注嵌入式)

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人

都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

  • 20
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值