珠海网站设计费用,网页制作个人简历网页的步骤,北京网上注册公司,用书籍上的文章做网站SEO1、字符型驱动设备你是怎么创建设备文件的#xff0c;就是/dev/下面的设备文件#xff0c;供上层应用程序打开使用的文件#xff1f;
答#xff1a;mknod命令结合设备的主设备号和次设备号#xff0c;可创建一个设备文件。
评#xff1a;这只是其中一种方式#xff0…1、字符型驱动设备你是怎么创建设备文件的就是/dev/下面的设备文件供上层应用程序打开使用的文件
答mknod命令结合设备的主设备号和次设备号可创建一个设备文件。
评这只是其中一种方式也叫手动创建设备文件。还有UDEV/MDEV自动创建设备文件的方式UDEV/MDEV是运行在用户态的程序可以动态管理设备文件包括创建和删除设备文件运行在用户态意味着系统要运行之后。那么在系统启动期间还有devfs创建了设备文件。一共有三种方式可以创建设备文件。 2、写一个中断服务需要注意哪些如果中断产生之后要做比较多的事情你是怎么做的 答中断处理例程应该尽量短把能放在后半段(tasklet等待队列等)的任务尽量放在后半段。 评写一个中断服务程序要注意快进快出在中断服务程序里面尽量快速采集信息包括硬件信息然后推出中断要做其它事情可以使用工作队列或者tasklet方式。也就是中断上半部和下半部。 第二中断服务程序中不能有阻塞操作。为什么大家可以讨论。
第三中断服务程序注意返回值要用操作系统定义的宏做为返回值而不是自己定义的OKFAIL之类的。 3、自旋锁和信号量在互斥使用时需要注意哪些在中断服务程序里面的互斥是使用自旋锁还是信号量还是两者都能用为什么 答使用自旋锁的进程不能睡眠使用信号量的进程可以睡眠。中断服务例程中的互斥使用的是自旋锁原因是在中断处理例程中硬中断是关闭的这样会丢失可能到来的中断。 4、原子操作你怎么理解为了实现一个互斥自己定义一个变量作为标记来作为一个资源只有一个使用者行不行 答原子操作指的是无法被打断的操作。我没懂第二句是什么意思自己定义一个变量怎么可能标记资源的使用情况其他进程又看不见这个变量 评第二句话的意思是 定义一个变量,比如 int flag 0; if(flag 0) { flag 1; 操作临界区 flag 0; }这样可否 5、insmod 一个驱动模块会执行模块中的哪个函数rmmod呢这两个函数在设计上要注意哪些遇到过卸载驱动出现异常没是什么问题引起的 答insmod调用init函数rmmod调用exit函数。这两个函数在设计时要注意什么卸载模块时曾出现卸载失败的情形原因是存在进程正在使用模块检查代码后发现产生了死锁的问题。
评要注意在init函数中申请的资源在exit函数中要释放包括存储ioremap定时器工作队列等等。也就是一个模块注册进内核退出内核时要清理所带来的影响带走一切不留下一点痕迹。 6、在驱动调试过程中遇到过oops没你是怎么处理的 7、ioctl和unlock_ioctl有什么区别 8、驱动中操作物理绝对地址为什么要先ioremap? 答因为内核没有办法直接访问物理内存地址必须先通过ioremap获得对应的虚拟地址。
9、设备驱动模型三个重要成员是platfoem总线的匹配规则是在具体应用上要不要先注册驱动再注册设备有先后顺序没 10、Linux中内核空间及用户空间的区别用户空间与内核通信方式有哪些? 11、linux中内存划分及如何使用虚拟地址及物理地址的概念及彼此之间的转化高端内存概念 12、linux中中断的实现机制tasklet与workqueue的区别及底层实现区别为什么要区分上半部和下半部 13、linux中断的响应执行流程中断的申请及何时执行(何时执行中断处理函数) 14、linux中的同步机制spinlock与信号量的区别 15、linux中RCU原理 16、linux中软中断的实现原理 17、linux系统实现原子操作有哪些方法 18、MIPS Cpu中空间地址是怎么划分的如在uboot中如何操作设备的特定的寄存器 19、linux中系统调用过程如:应用程序中read()在linux中执行过程即从用户空间到内核空间 20、linux内核的启动过程(源代码级) 21、linux调度原理 22、linux网络子系统的认识 23、linux内核里面内存申请有哪几个函数各自的区别 Kmalloc() __get_free_page() mempool_create() 24. IRQ和FIQ有什么区别在CPU里面是是怎么做的 25. 中断的上半部分和下半部分的问题讲下分成上半部分和下半部分的原因为何要分讲下如何实现 上半部分执行与硬件相关的处理要求快, 而有些驱动在中断处理程序中又需要完成大量工作,这构成矛盾,所以Linux有所谓的bottom half机制中断处理程序中所有不要求立即完成的,在开中断的环境下,由底半程序随后完成. Linux的底半处理实际上是建立在内核的软中断机制上的. Linux 的底半 机制主要有Tasklet 和 work queue 以及 softirq ( 2.4内核则有BH , Task queue , softirq , tasklet 没有work queue)其实底半可以理解成一种工作的延迟。所以实际使用时跟timer机制基本上一个意思。 26. 内核函数mmap的实现原理机制 mmap函数实现把一个文件映射到一个内存区域从而我们可以像读写内存一样读写文件他比单纯调用read/write也要快上许多。在某些时候我们可以把内存的内容拷贝到一个文件中实现内存备份当然也可以把文件的内容映射到内存来恢复某些服务。另外mmap实现共享内存也是其主要应用之一mmap系统调用使得进程之间通过映射同一个普通文件实现共享内存。 27. 驱动里面为什么要有并发、互斥的控制如何实现讲个例子 28. spinlock自旋锁是如何实现的 自旋锁在同一时刻只能被最多一个内核任务持有所以一个时刻只有一个线程允许存在于临界区中。这点可以应用在多处理机器、或运行在单处理器上的抢占式内核中需要的锁定服务。 这里也介绍下信号量的概念因为它的用法和自旋锁有相似的地方。linux中的信号量是一种睡眠锁。如果有一个任务试图获得一个已被持有的信号量时信号量会将其推入等待队列然后让其睡眠。这时处理器获得自由去执行其它代码。当持有信号量的进程将信号量释放后在等待队列中的一个任务将被唤醒从而便可以获得这个信号量。 29. 任务调度的机制 30. 嵌入式linux和wince操作系统的特点和特性 31. 嵌入式linux中tty设备驱动的体系结构 32. 嵌入式设备为加快启动速度可以做哪些方面的优化 33. USB设备的枚举过程 (1) Get Device Descriptor。主机的第一个命令要求得到设备描述符此SETUP 包为8 个字节数据8006000100004000发向地址0端口0。“40”表示返回数据长度最大为40H 个字节。实际上只返回一个包即数组DEV_DESC[ ]中的前8 个字节用于说明设备的描述符的真实长度和设备的类型。 (2) Set Address。接着是设置设备地址处理事件主机发送一个含有指定地址的数据包0005020000000000在主机只有一个USB 设备的时候这个地址一般会是2最大地址127USB 协议中可以连接127 个设备。设置地址事件处理结束后设备进入地址状态主机以后会在新的指定地址处访问设备。 (3) Get Device Descriptor。主机再次发送请求得到设备描述符的数据包8006000100001200与上次不同的是要求的数据的长度是实际的数据长度同时是发送到Set Address命令所设置的地址。 (4) 读取全部Configuration Descriptor。接着主机要求得到设备全部的配置描述符、接口描述符和节点描述符8006000200004000由于主机不知道设备描述符的真实长度因此它要求得到64个字节。 (5) Set Interface主机发送数据包010B000000000000设置接口值为0。 (6) Set Conifguration确定USB设备工作在哪一个配置下。对于U盘设备来说一般只有1个配置值其值为01。主机发送数据包0009010000000000。 (7) 如果以上步骤都正确主机将找到新设备并且配置成功该设备可以正常使用可以进行后续的U盘枚举过程了。 (8) 用busHound观察计算机对于U盘的枚举过程发现上述步骤后还有一个GetMaxLun的操作但是实际上对于U盘来说忽略该步骤也没有问题。 34. PSRAM、SDRAM、DDR、DDR2的时序特性
35.什么是GPIO general purpose input/output GPIO是相对于芯片本身而言的如某个管脚是芯片的GPIO脚则该脚可作为输入或输出高或低电平使用当然某个脚具有复用的功能即可做GPIO也可做其他用途。 也就是说你可以把这些引脚拿来用作任何一般用途的输入输出例如用一根引脚连到led的一极来控制它的亮灭也可以用一根一些引脚连到一个传感器上以获得该传感器的状态这给cpu提供了一个方便的控制周边设备的途经。如果没有足够多的gpio管脚在控制一些外围设备时就会力有不逮这时可采取的方案是使用CPLD来帮助管理。 36.触摸屏的硬件原理 触摸屏的主要三大种类是电阻技术触摸屏、 表面声波技术触摸屏、 电容技术触摸屏。 电阻触摸屏的主要部分是一块与显示器表面非常配合的电阻薄膜屏 这是一种多层的复合薄膜它以一层玻璃或硬塑料平板作为基层表面图有一层透明氧化金属 ITO氧化铟透明的导电电阻 导电层上面在盖有一层外表面硬化处理、光滑防擦的塑料层 、它的内表面也涂有一层ITO涂层 、在他们之间有许多细小的小于1/1000英寸的透明隔离点把两层导电层隔开绝缘 。当手指触摸屏幕时两层导电层在触摸点位置就有了接触控制器侦测到这一接触并计算出XY 的位置再根据模拟鼠标的方式运作。这就是电阻技术触摸屏的最基本的原理。 表面声波技术是利用声波在物体的表面进行传输当有物体触摸到表面时阻碍声波的传输换能器侦测到这个变化反映给计算机进而进行鼠标的模拟。 电容技术触摸屏利用人体的电流感应进行工作 。用户触摸屏幕时 由于人体电场用户和触摸屏表面形成以一个耦合电容 对于高频电流来说电容是直接导体于是手指从接触点吸走一个很小的电流 37.在Linux C中ls这个命令是怎么被执行的? 使用fork创建一个进程或exec函数族覆盖原进程。 38.在一个只有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); } 第一道程序分配内存但没有填充编译器可能会把内存分配优化掉程序死循环第二道程序分配内存并进行填充系统会一直分配内存直到内存不足退出循环。 39.请定义一个宏比较两个数a、b的大小不能使用大于、小于、if语句 搞的比较复杂。主要思想就是a-b的值的最高位是否为0但是又得考虑整数溢出的问题所以很复杂。不知道哪位大侠有更好的办法指点指点。 #includestdio.h #define ZHENG(i)((i 31) 0) #define FU(i)((i 31)! 0) #define COMPARE(a,b)((ZHENG(a) FU(b))||(((ZHENG(a) ZHENG(b))||(FU(a)FU(b)))((((a)-(b)) 31) 0))) void main() { int a 0x80000001; int b 0x6FFFFFFF; if(COMPARE(a,b)) { printf(a b\n); } else { printf(a b\n); } } 40、LINUX下的Socket套接字和Windows下的WinSock有什么共同点请从C/C语言开发的角度描述至少说出两点共同点。 参考答案 第1题答中一个得5分答出其它正确答案的也得5分。 a)都基于TCP/IP协议都提供了面向连接的TCP SOCK和无连接的UDP SOCK。 b)都是一个sock结构体。 c)都是使用sock文件句柄进行访问。 d)都具有缓冲机制。 41、请编写一个标准Shell脚本testd实现如下功能 A、在Linux操作系统启动的时候自动加载/mnt/test/test程序。 B、当test异常退出之后自动重新启动。 C、当test程序重启次数超过100次自动复位操作系统。 假设你所拥有的资源 A、目标机器是一台具有标准shell的嵌入式计算机CPU为ARM7 56MB内存16MB软件环境基于Linux2.6.11和BusyBox1.2构建。 B、当前已有11个用户进程在运行占用了大部分的CPU时间和内存你可使用的内存只有2MB左右CPU时间由系统分派。 本题是考查LINUX和嵌入式编程功底的写出程序来的不少但是95%以上的人竟无视我假设的资源不知道在重启test程序的时候需要加上一个适当的掩饰时间以便资源紧张的操作系统有时间回收资源。85%的人不知道写完testd之后要在init里边加载这个脚本才能实现启动时自动加载的功能。 参考答案 ######################################## #testd is a daemon script to start an watch the program test ######################################## #!/bin/sh #load *.so that may need if [ -r /sbin/ldconfig ]; then ldconfig fi #add the libs PATH that may need export LD_LIBRARY_PATH/lib #count is the counter of test started times count0 #main loop while [ 1 ] ;do #add execute property for /mnt/test/test chmod x /mnt/test/test #start test /mnt/test/test #the running times counter let countcount1 echo test running times is $count #Is test running too many times? if [ $count -gt 100 ]; then echo Will reboot because of test running too many times reboot fi #wait for test stoping...
sleep 3 done ######################################### 42.你平常是怎么用C写嵌入式系统的死循环的? 43.写一条命令实现在dir以及其子目录下找出所有包含“hello world”字符串的文件 44.下面的两段程序中循环能否执行为什么 A: unsigned short i; unsigned short index 0; for(i 0; i index-1; i){ printf(“a\n”); } B: unsigned short i; unsigned long index 0; for(i 0; i index-1; i){ printf(“b\n”); } 45.一个计划跑LINUX系统的ARM系统把bootloader烧录进去后上电后串口上没有任何输出硬件和软件各应该去检查什么 提示 1.跑LINUX的系统一般都需要外扩DRAM,一般的系统也经常有NOR或NAND FLASH 476列举最少3种你所知道的嵌入式的体系结构并请说明什么是ARM体系结构。 47.请简述下面这段代码的功能
mov r12, #0x0
ldr r13, 0x30100000
mov r14, #4096
loop:
ldmia r12!, {r0-r11}
stmia r13!, {r0-r11}
cmp r12, r14
bl loop
48.嵌入式中常用的文件系统有哪些说出它们的主要特点和应用场合? 49.某外设寄存器rGpioBase的地址是0x56000000,寄存器的0~15位有效请写出给外设寄存器高八位(8~15位)设置成0xc3的代码 50.如何编写一个LINUX驱动 提示主要说字符设备的编写过程 51.简述LINUX驱动中字符设备和块设备的区别 52.试总结单片机底层开发与LINUX驱动开发有哪些异同 53.请从网卡、USB HOST、LCD驱动器、NAND FLASH、WIFI 、音频芯片中选择一个或者2个可以以具体的芯片为例对下面的问题做答 1)如果是外部扩展芯片请说出你用的芯片的型号 2)画出上题中你选定相应硬件模块与CPU的主要引脚连线
3) 编写上题中你选定相应硬件模块相应LINUX驱动的流程 54、linux驱动分类
Linux设备驱动的分类 1字符设备。 2 块设备。 3 网络设备。 字符设备指那些必须以串行顺序依次进行访问的设备如触摸屏、磁带驱动器、鼠标等。块设备可以用任意顺序进行访问以块为单位进行操作如硬盘、软驱等。字符设备不经过系统的快速缓冲而块设备经过系统的快速缓冲。但是字符设备和块设备并没有明显的界限如对于Flash设备符合块设备的特点但是我们仍然可以把它作为一个字符设备来访问。网络设备在Linux里做专门的处理。Linux的网络系统主要是基于BSD unix的socket 机制。在系统和驱动程序之间定义有专门的数据结构(sk_buff)进行数据的传递。系统里支持对发送数据和接收数据的缓存提供流量控制机制提供对多协议的支持。 55、信号量与自旋锁
自旋锁 自旋锁是专为防止多处理器并发而引入的一种锁它应用于中断处理等部分。对于单处理器来说防止中断处理中的并发可简单采用关闭中断的方式不需要自旋锁。 自旋锁最多只能被一个内核任务持有如果一个内核任务试图请求一个已被争用(已经被持有)的自旋锁那么这个任务就会一直进行忙循环——旋转——等待锁重新可用。要是锁未被争用请求它的内核任务便能立刻得到它并且继续进行。自旋锁可以在任何时刻防止多于一个的内核任务同时进入临界区因此这种锁可有效地避免多处理器上并发运行的内核任务竞争共享资源。 事实上自旋锁的初衷就是在短期间内进行轻量级的锁定。一个被争用的自旋锁使得请求它的线程在等待锁重新可用的期间进行自旋(特别浪费处理器时间)所以自旋锁不应该被持有时间过长。如果需要长时间锁定的话, 最好使用信号量。但是自旋锁节省了上下文切换的开销。 自旋锁的基本形式如下 spin_lock(mr_lock); //临界区 spin_unlock(mr_lock); 因为自旋锁在同一时刻只能被最多一个内核任务持有所以一个时刻只有一个线程允许存在于临界区中。这点很好地满足了对称多处理机器需要的锁定服务。在单处理器上自旋锁仅仅当作一个设置内核抢占的开关。如果内核抢占也不存在那么自旋锁会在编译时被完全剔除出内核。 简单的说自旋锁在内核中主要用来防止多处理器中并发访问临界区防止内核抢占造成的竞争。另外自旋锁不允许任务睡眠(持有自旋锁的任务睡眠会造成自死锁——因为睡眠有可能造成持有锁的内核任务被重新调度而再次申请自己已持有的锁)它能够在中断上下文中使用。 死锁假设有一个或多个内核任务和一个或多个资源每个内核都在等待其中的一个资源但所有的资源都已经被占用了。这便会发生所有内核任务都在相互等待但它们永远不会释放已经占有的资源于是任何内核任务都无法获得所需要的资源无法继续运行这便意味着死锁发生了。自死琐是说自己占有了某个资源然后自己又申请自己已占有的资源显然不可能再获得该资源因此就自缚手脚了。递归使用一个自旋锁就会出现这种情况。 信号量 信号量是一种睡眠锁。如果有一个任务试图获得一个已被持有的信号量时信号量会将其推入等待队列然后让其睡眠。这时处理器获得自由去执行其它代码。当持有信号量的进程将信号量释放后在等待队列中的一个任务将被唤醒从而便可以获得这个信号量。 信号量的睡眠特性使得信号量适用于锁会被长时间持有的情况只能在进程上下文中使用因为中断上下文中是不能被调度的另外当代码持有信号量时不可以再持有自旋锁。 信号量基本使用形式为 static DECLARE_MUTEX(mr_sem);//声明互斥信号量 if(down_interruptible(mr_sem)) //可被中断的睡眠当信号来到睡眠的任务被唤醒 //临界区 up(mr_sem); 信号量和自旋锁区别 从严格意义上讲信号量和自旋锁属于不同层次的互斥手段前者的实现有赖于后者。 注意以下原则: 如果代码需要睡眠——这往往是发生在和用户空间同步时——使用信号量是唯一的选择。由于不受睡眠的限制使用信号量通常来说更加简单一些。如果需要在自旋锁和信号量中作选择应该取决于锁被持有的时间长短。理想情况是所有的锁都应该尽可能短的被持有但是如果锁的持有时间较长的话使用信号量是更好的选择。另外信号量不同于自旋锁它不会关闭内核抢占所以持有信号量的代码可以被抢占。这意味者信号量不会对影响调度反应时间带来负面影响。 自旋锁对信号量 需求 建议的加锁方法 低开销加锁 优先使用自旋锁 短期锁定 优先使用自旋锁 长期加锁 优先使用信号量 中断上下文中加锁 使用自旋锁 持有锁是需要睡眠、调度 使用信号量 56、platform总线设备及总线设备如何编写 57、kmalloc和vmalloc的区别
kmalloc()和vmalloc()介绍 kmalloc() 用于申请较小的、连续的物理内存 1. 以字节为单位进行分配在linux/slab.h中 2. void *kmalloc(size_t size, int flags) 分配的内存物理地址上连续虚拟地址上自然连续 3. gfp_mask标志什么时候使用哪种标志如下 ———————————————————————————————- 情形 相应标志 ———————————————————————————————- 进程上下文可以睡眠 GFP_KERNEL 进程上下文不可以睡眠 GFP_ATOMIC 中断处理程序 GFP_ATOMIC 软中断 GFP_ATOMIC Tasklet GFP_ATOMIC 用于DMA的内存可以睡眠 GFP_DMA | GFP_KERNEL 用于DMA的内存不可以睡眠 GFP_DMA | GFP_ATOMIC ———————————————————————————————- 4. void kfree(const void *ptr) 释放由kmalloc()分配出来的内存块 vmalloc() 用于申请较大的内存空间虚拟内存是连续的 1. 以字节为单位进行分配在linux/vmalloc.h中 2. void *vmalloc(unsigned long size) 分配的内存虚拟地址上连续物理地址不连续 3. 一般情况下只有硬件设备才需要物理地址连续的内存因为硬件设备往往存在于MMU之外根本不了解虚拟地址但为了性能上的考虑内核中一般使用 kmalloc()而只有在需要获得大块内存时才使用vmalloc()例如当模块被动态加载到内核当中时就把模块装载到由vmalloc()分配 的内存上。 4.void vfree(void *addr)这个函数可以睡眠因此不能从中断上下文调用。 malloc(), vmalloc()和kmalloc()区别 [*]kmalloc和vmalloc是分配的是内核的内存,malloc分配的是用户的内存 [*]kmalloc保证分配的内存在物理上是连续的,vmalloc保证的是在虚拟地址空间上的连续,malloc不保证任何东西(这点是自己猜测的,不一定正确) [*]kmalloc能分配的大小有限,vmalloc和malloc能分配的大小相对较大 [*]内存只有在要被DMA访问的时候才需要物理上连续 [*]vmalloc比kmalloc要慢 58、module_init的级别 59、添加驱动
静态加载和动态加载 静态加载是系统启动的时候由内核自动加载的这个要事先将驱动编译进内核才行 动态加载也就是模块加载方式这种方式下驱动以模块的形式存放在文件系统中需要时动态载入内核这种主要用在调试的时候比较方便灵活。insmod module.ko 60、IIC原理总线框架设备编写方法i2c_msg 61、kernel panic 62、USB总线USB传输种类urb等
USB总线 USB总线属于一种轮询式总线主机控制端口初始化所有的数据传输。每一总线动作最多传送三个数据包包括令牌(Token)、数据(Data)、联络(HandShake)。按照传输前制定好的原则在每次传送开始时主机送一个描述传输动作的种类、方向、USB设备地址和终端号的USB数据包这个数据包通常被称为令牌包(TokenPacket)。USB设备从解码后的数据包的适当位置取出属于自己的数据。数据传输方向不是从主机到设备就是从设备到主机。在传输开始时由标志包来标志数据的传输方向然后发送端开始发送包含信息的数据包或表明没有数据传送。接收端也要相应发送一个握手的数据包表明是否传送成功。发送端和接收端之间的USB数据传输在主机和设备的端口之间可视为一个通道。USB中有一个特殊的通道一缺省控制通道它属于消息通道设备一启动即存在从而为设备的设置、状态查询和输入控制信息提供一个入口。 USB总线的四种传输类型 1、中断传输由OUT事务和IN事务构成用于键盘、鼠标等HID设备的数据传输中 2、批量传输由OUT事务和IN事务构成用于大容量数据传输没有固定的传输速率也不占用带宽当总线忙时USB会优先进行其他类型的数据传输而暂时停止批量转输。 3、同步传输由OUT事务和IN事务构成有两个特别地方第一在同步传输的IN和OUT事务中是没有返回包阶段的第二在数据包阶段任何的数据包都为DATA0 4、控制传输最重要的也是最复杂的传输控制传输由三个阶段构成初始配置阶段、可选数据阶段、状态信息步骤每一个阶段能够看成一个的传输也就是说控制传输其实是由三个传输构成的用来于USB设备初次加接到主机之后主机通过控制传输来交换信息设备地址和读取设备的描述符使得主机识别设备并安装相应的驱动程式这是每一个USB研发者都要关心的问题。 URB: USB请求块USB request blockurb是USB设备驱动中用来描述与USB设备通信所用的基本载体和核心数据结构非常类似于网络设备驱动中的sk_buff结构体是USB主机与设备通信的“电波”。 63、同步和互斥 同步和互斥 相交进程之间的关系主要有两种同步与互斥。所谓互斥是指散步在不同进程之间的若干程序片断当某个进程运行其中一个程序片段时其它进程就不能运行它们之中的任一程序片段只能等到该进程运行完这个程序片段后才可以运行。所谓同步是指散步在不同进程之间的若干程序片断它们的运行必须严格按照规定的某种先后次序来运行这种先后次序依赖于要完成的特定的任务。 显然同步是一种更为复杂的互斥而互斥是一种特殊的同步。也就是说互斥是两个线程之间不可以同时运行他们会相互排斥必须等待一个线程运行完毕另一个才能运行而同步也是不能同时运行但他是必须要安照某种次序来运行相应的线程也是一种互斥 总结: 互斥是指某一资源同时只允许一个访问者对其进行访问具有唯一性和排它性。但互斥无法限制访问者对资源的访问顺序即访问是无序的。 同步是指在互斥的基础上大多数情况通过其它机制实现访问者对资源的有序访问。在大多数情况下同步已经实现了互斥特别是所有写入资源的情况必定是互斥的。少数情况是指可以允许多个访问者同时访问资源
64、 Linux设备中字符设备与块设备有什么主要的区别请分别列举一些实际的设备说出它们是属于哪一类设备。 答字符设备字符设备是个能够像字节流类似文件一样被访问的设备由字符设备驱动程序来实现这种特性。字符设备驱动程序通常至少实现open,close,read和write系统调用。字符终端、串口、鼠标、键盘、摄像头、声卡和显卡等就是典型的字符设备。 块设备和字符设备类似块设备也是通过/dev目录下的文件系统节点来访问。块设备上能够容纳文件系统如u盘SD卡磁盘等。 字符设备和块设备的区别仅仅在于内核内部管理数据的方式也就是内核及驱动程序之间的软件接口而这些不同对用户来讲是透明的。在内核中和字符驱动程序相比块驱动程序具有完全不同的接口 65、查看驱动模块中打印信息应该使用什么命令如何查看内核中已有的字符设备的信息如何查看正在使用的有哪些中断号 答1) 查看驱动模块中打印信息的命令dmesg 2) 查看字符设备信息可以用lsmod 和modprobelsmod可以查看模块的依赖关系modprobe在加载模块时会加载其他依赖的模块。 3显示当前使用的中断号cat /proc/interrupt 66、Linux中引入模块机制有什么好处 答首先模块是预先注册自己以便服务于将来的某个请求然后他的初始化函数就立即结束。换句话说模块初始化函数的任务就是为以后调用函数预先作准备。 好处 1) 应用程序在退出时可以不管资源的释放或者其他的清除工作但是模块的退出函数却必须仔细此撤销初始化函数所作的一切。 2) 该机制有助于缩短模块的开发周期。即注册和卸载都很灵活方便。 67、copy_to_user()和copy_from_user()主要用于实现什么功能一般用于file_operations结构的哪些函数里面 答由于内核空间和用户空间是不能互相访问的如果需要访问就必须借助内核函数进行数据读写。copy_to_user():完成内核空间到用户空间的复制copy_from_user()是完成用户空间到内核空间的复制。一般用于file_operations结构里的read,write,ioctl等内存数据交换作用的函数。当然如果ioctl没有用到内存数据复制那么就不会用到这两个函数。
68、请简述主设备号和次设备号的用途。如果执行mknod chartest c 4 64创建chartest设备。请分析chartest使用的是那一类设备驱动程序。 答 1主设备号主设备号标识设备对应的驱动程序。虽然现代的linux内核允许多个驱动程序共享主设备号但我们看待的大多数设备仍然按照“一个主设备对应一个驱动程序”的原则组织。 次设备号次设备号由内核使用用于正确确定设备文件所指的设备。依赖于驱动程序的编写方式我们可以通过次设备号获得一个指向内核设备的直接指针也可将此设备号当作设备本地数组的索引。 2chartest 由驱动程序4管理该文件所指的设备是64号设备。感觉类似于串口终端或者字符设备终端。 69、设备驱动程序中如何注册一个字符设备分别解释一下它的几个参数的含义。 答注册一个字符设备驱动有两种方法 1 void cdev_init(struct cdev *cdev, struct file_operations *fops) 该注册函数可以将cdev结构嵌入到自己的设备特定的结构中。cdev是一个指向结构体cdev的指针而fops是指向一个类似于file_operations结构可以是file_operations结构但不限于该结构的指针. 2 int register_chrdev(unsigned int major, const char *namem , struct file)operations *fopen); 该注册函数是早期的注册函数major是设备的主设备号name是驱动程序的名称而fops是默认的file_operations结构这是只限于file_operations结构。对于register_chrdev的调用将为给定的主设备号注册0255作为次设备号并为每个设备建立一个对应的默认cdev结构。 70、请简述中断与DMA的区别。Linux设备驱动程序中使用哪个函数注册和注销中断处理程序 答1DMA是一种无须CPU的参与就可以让外设与系统内存之间进行双向数据传输的硬件机制使用DMA可以使系统CPU从实际的IO数据传输过程中摆脱出来从而大大提高系统的吞吐率。 中断是指CPU在执行程序的过程中出现了某些突发事件时CPU必须暂停执行当前的程序转去处理突发事件处理完毕后CPU又返回源程序被中断的位置并继续执行。 所以中断和MDA的区别就是MDA不需CPU参与而中断是需要CPU参与的。 2中断注册函数和中断注销函数 注册中断 int request_irq(unsigned int irq, irqreturn_t (*handler)(int, void *, struct pt_regs *), unsigned long flags, const char *dev_name, void *dev_id); 参数意义依次是中断号中断处理函数中断管理有关的掩码中断请求设备名中断信号线。 过程是dev_name设备请求中断-cpu分配中断号-设置中断管理的掩码-分配中断信号线-处理中断函数-完成之后再根据设置情况返回原处理程序处继续处理程序。 注销中断 Void free_irq(unsigned int irq, void *dev_id); 释放中断和中断信号线 71、中断和轮询哪个效率高怎样决定是采用中断方式还是采用轮询方式去实现驱动 答中断是CPU处于被中状态下来接受设备的信号而轮询是CPU主动去查询该设备是否有请求。凡事都是两面性所以看效率不能简单的说那个效率高。如果是请求设备是一个频繁请求cpu的设备或者有大量数据请求的网络设备那么轮询的效率是比中断高。如果是一般设备并且该设备请求cpu的频率比较底则用中断效率要高一些。 72、简单描述在cs8900的驱动设计中, 发送数据frame和接收数据frame的过程。 答1发送流程如下 1 网络设备驱动程序从上层协议传递过来的sk_buff参数获得数据包的有效数据和长度将有效数据放入临时缓冲区。 2 对于以太网如果有效数据的长度小于以太网冲突检测所要求的数据桢的最小长度则给临时缓冲区的末尾填充0 3 设置硬件寄存器驱使网络设备进行数据发送操作。 2接收流程 网络设备接收数据主要是由中断引发设备的中断处理函数中断处理函数判断中断类型如果为接收中断则读取接受到的数据分配sk_buff数据结构和数据缓冲区将接收到的数据复制到数据缓冲区并调用netif_rx()函数将sk_buff传递给上层协议。 73、cs8900.c的驱动中发送数据frame的过程为什么需要关中断?接收数据frame的过程为什么不需要关中断? 答在发送过程中是不能被打断的在发送的过程中不关中断这时候如果有一个中断到来那么cpu有可能会去相应该中断如果该中断需要改写的数据是发送数据的缓冲区那么缓冲区将被改写这样即使cpu相应完毕该中断再发送数据接收方也不认识该数据不能接收。 在接收数据的时候需要打开中断是因为要及时的相应接收到的数据。如果关闭该中断那么接收方有可能因为相应优先级高的中断而接收不到该数据。 74、简单描述skbuff这个数据结构在网络结构中所起到的作用,为什么需要一个skbuff,它的分配和释放主要都在什么部位 答sk_buff结构非常重要它的含义为“套接字缓冲区”用于在linux网络子系统中的各层之间传递数据。 当发送数据包时linux内核的网络处理模块必须建立一个包含要传输的数据包的sk_buff然后将sk_buff递交给下层各层在sk_buff中添加不同的协议头直至交给网络设备发送。同样的当网络设备从网络媒介上接受到数据包后它必须将接受到的数据转换为sk_buff数据结构并传递给上层盖层不抛去相应的协议头直至交给用户。分配sk_buff在接收一开始就应该分配在发送完毕数据之后可以释放sk_buff。