PCI总线常见于x86体系,本文默认面向的体系为x86,注意x86架构下IO与内存是独立编址的。 附: 本文默认读者熟悉Linux设备驱动模型,不熟悉的可以先阅读这两篇blog。...Linux驱动之I2C子系统剖析 Linux驱动之SPI子系统剖析 PCI寻址 PCI系统总体布局组织为树状,从CPU连接的Host Bridge引出PCI主桥,主桥连接的是PCI总线0,可以直接连接PCI...Linux内核启动时会从PCI设备的配置寄存器里读取内存/IO起始地址以及irq,并把这些信息赋值给struct pci_dev的相应成员来生成软件描述的PCI设备。...val) int pci_write_config_dword(struct pci_dev *dev, int where, u32 val) PCI驱动的注册及匹配 BIOS在启动时,会为每个PCI...当linux系统启动时,会探测系统中的所有PCI设备,并为探测到的每个PCI设备做如下操作: 1.分配一个struct pci_dev结构体,用来表示相应的PCI设备 2.为这个结构体填充设备vendor
由于实在找不到MSI-X在x86上实现的教程或文档,只能分析Linux的PCI驱动程序了。希望能得到一些启发。...本文基于linux 5.17.5进行分析 __pci_enable_msix_range static int __pci_enable_msix_range(struct pci_dev *dev,...该函数原型如下: int pci_msi_setup_msi_irqs(struct pci_dev *dev, int nvec, int type) 该函数首先获取msi的domain,校验是否为继承的...(我还不明白这里是干啥意思的) 然后就会去调用pci_msi_legacy_setup_msi_irqs函数,去设置msix。...pci_msi_legacy_setup_msi_irqs 该函数的作用很简单:调用arch_setup_msi_irqs方法,去设置msi,然后通过pci_msi_setup_check_result
其实,只要你认真下去,虽然有些东西看不明白,但是对于你写PCI的驱动来说,似乎“不那么重要”。因为,Linux内核对PCI总线已经有了完美的支持,你所需要做的内容是非常小的一部份。...众所周知,Linux 2.6内核引入了总线驱动模型这一概念,如此,很多基于总线的设备驱动就分成了总线驱动和设备驱动两部分。...下面是我写的一个PCI总线的驱动程序,注意是PCI设备识别时的驱动程序,这里并没有实现具体的功能驱动。...基于PCI总线的设备有很多种,但就PCI总线驱动这一块来说,都大同小异,实现了PCI总线驱动之后,再去继续做具体的设备驱动。...> #include linux/pci.h> #include linux/init.h> #include linux/delay.h> #include #include
pcibios_init x86 BIOS专门提供了针对PCI总线的操作,这些操作里就包括了总线枚举的整个过程,Linux kernel中的宏CONFIG_PCI_BIOS。...在系统加电以后自检时,就会完成对PCI总线的枚举,之后Linux对PCI配置空间的访问都是通过BIOS调用的形式进行,提供有这些功能和服务的BIOS就称之为PCI BIOS 。...需要注意的是Linux x86_64是不采用PCI BIOS访问PCI配置空间的,而是内核实现了直接访问PCI配置空间的函数(CONFIG_PCI_DIRECT)。...所以Linux x86驱动程序中pci_read_config_byte()最终调用的是pci_bios_read_config_byte()。...Android libpci库 external/pciutils 5.2 libpci判断一个PCI设备是不是PCIe capability ID参考:include/uapi/linux/pci_regs.h
Linux系统一次读取磁盘的大小是一个块,而不是一个扇区,块设备驱动由此得名。 二、块设备处理过程 1、linux 内核中,块设备将数据存储与固定的大小的块中,每个块都有自己的固定地址。...Linux内核中块设备和其他模块的关系如下。 ? 1、块设备的处理过程涉及Linux内核中的很多模块,下面简单描述之间的处理过过程。 ...linux 块设备驱动架构图> ?...(Linux系统中,对块设备的IO请求,都会向块设备驱动发出一个请求,在驱动中用request结构体描述) 内核结构如下:. struct request { struct list_head queuelist...(也可以不用注册设备,驱动一样可以工作,该函数和字符设备的register_chrdev()函数相对应,对于大多数的块设备,第一个工作就是相内核注册自己,但是在Linux2.6以后,register_blkdev
转载请标明出处floater的csdn blog,http://blog.csdn.net/flaoter Linux SPI驱动分为核心层,控制器驱动层和设备驱动层。...核心层是Linux的SPI核心部分,提供了核心数据结构的定义,总线、设备和驱动的注册、注销管理等,提供与上层的统一接口。...linux将I2C、SPI、USB等总线驱动隔离成控制器驱动和设备驱动,使两者相对独立。 本文以qcom的spi控制器为例,对spi控制器驱动进行解析。kernel代码版本是3.18。...linux驱动与设备是一对多的关系,在spi_master设备注册时,控制器的结构体信息会提供给spi_master作为私有数据。...3.3 厂商的总线传输 控制器驱动注册函数中进行了传输函数的赋值。
应用层可以借用这些接口访问挂接在适配器上的I2C设备的存储空间或寄存器, 并控制I2C设备的工作方式 显然,它和前几次驱动类似, I2C也分为总线驱动和设备驱动,总线就是协议相关的,它知道如何收发数据...,但不知道数据含义,设备驱动却知道数据含义 1.2 I2C驱动架构,如下图所示: ? ...i2c_add_adapter()和i2c_add_numbered_adapter().由于在系统中可能存在多个adapter,因为将每一条I2C总线对应一个编号,下文中称为I2C总线号.这个总线号的PCI...#include linux/kernel.h> #include linux/init.h> #include linux/module.h> #include linux/slab.h> #...include linux/jiffies.h> #include linux/i2c.h> #include linux/mutex.h> #include linux/fs.h> #include
/module.h> #include linux/kernel.h> #include linux/errno.h> #include linux/string.h> #include linux.../mm.h> #include linux/slab.h> #include linux/delay.h> #include linux/fb.h> #include linux/init.h>...4.测试运行 测试有两种: (echo和cat命令详解入口地址: http://www.cnblogs.com/lifexy/p/7601122.html) echo hello> /dev/tty1... // LCD上便显示hello字段 cat Makefile>/dev/tty1 // LCD上便显示Makeflie文件的内容 4.1使用上节的键盘驱动在LCD终端运行linux vi...如下图,我们insmod上一节的键盘驱动后,按下enter键,便能在LCD终端上操作linux了 (上一节的键盘驱动详解入口地址: http://www.cnblogs.com/lifexy/p/7553861
上一节 我们学习了: IIC接口下的24C02 驱动分析 接下来本节, 学习Linux下如何利用linux下I2C驱动体系结构来操作24C02 1....I2C体系结构分析 1.1首先进入linux内核的driver/i2c目录下,如下图所示: ?...i2c_add_adapter()和i2c_add_numbered_adapter().由于在系统中可能存在多个adapter,因为将每一条I2C总线对应一个编号,下文中称为I2C总线号.这个总线号的PCI...: /* * I2C-24C02 */ #include linux/kernel.h> #include linux/init.h> #include linux/module.h> #include...linux/slab.h> #include linux/jiffies.h> #include linux/i2c.h> #include linux/mutex.h> #include <
4)初始化定时器和中断 5)写中断服务函数 6)写定时器超时函数 7)在出口函数中 释放中断函数,删除定时器,卸载释放驱动 具体代码如下(都加了注释): #include linux/module.h...> #include linux/version.h> #include linux/init.h> #include linux/fs.h> #include linux/interrupt.h...linux/proc_fs.h> #include linux/delay.h> #include linux/platform_device.h> #include linux/input.h...,一种是直接打开/dev/tyy1,第二种是使用exec命令 (exec命令详解入口地址: http://www.cnblogs.com/lifexy/p/7553228.html) 方法1: cat...里面有关QT自启动的命令,然后重启 若板子没在QT下进行,也无法测试成功: 1)可以使用hexdump命令来调试代码 (hexdump命令调试代码详解地址:http://www.cnblogs.com
通过上节的块设备驱动分析,本节便通过内存来模拟块设备驱动 ,方便我们更加熟悉块设备驱动框架 参考内核自带的块设备驱动程序: drivers/block /xd.c drivers/block /z2ram.c.../module.h> #include linux/errno.h> #include linux/interrupt.h> #include linux/mm.h> #include linux.../fs.h> #include linux/kernel.h> #include linux/timer.h> #include linux/genhd.h> #include linux/hdreg.h...> #include linux/ioport.h> #include linux/init.h> #include linux/wait.h> #include linux/blkdev.h>...说明这个块设备测试运行无误 6.使用fdisk来对磁盘分区 (fdisk命令使用详解: http://www.cnblogs.com/lifexy/p/7661239.html) 共分了两个分区,如下图所示
对于2440来讲,因为此时的A0~A19的容量刚好为2MB,与cfi模式下读取的数据一致,所以没有接A20 3.接下来便来分析如何写norflash驱动 3.1 先来回忆下之前的nandflsh驱动:...驱动同样要实现硬件相关的操作,供给mtd设备调用 3.2参考内核自带的nor驱动:drivers/mtd/maps/physmap.c 进入它的init函数: 发现注册了两个platform平台设备驱动...linux/module.h> #include linux/types.h> #include linux/kernel.h> #include linux/init.h> #include...linux/slab.h> #include linux/device.h> #include linux/platform_device.h> #include linux/mtd/mtd.h...> #include linux/mtd/map.h> #include linux/mtd/partitions.h> #include static struct mtd_info
前言 随着计算机技术的飞速发展,Linux操作系统作为开源领域的佼佼者,已经深入到了各个应用场景之中。在Linux系统中,内核与用户空间之间的交互是核心功能之一,而设备驱动则是实现这一交互的关键环节。...然而,传统的设备驱动开发往往受限于内核空间的限制,无法充分发挥用户空间程序的灵活性和性能优势。为了解决这个问题,Linux内核引入了UIO(Userspace I/O)驱动模型。 ...本文将深入剖析Linux UIO驱动模型的技术细节,包括其定义、软件架构、必要性、工作原理以及涉及的内核函数等。...一、什么是UIO UIO(Userspace I/O)是Linux内核中的一个轻量级驱动框架,它允许用户空间程序直接访问物理设备资源,如内存、中断和DMA通道等。...六、结论 Linux UIO驱动模型提供了一种高效、灵活的方式来访问硬件设备资源。通过深入了解UIO驱动模型的架构、原理和内核函数,我们可以更好地利用这一模型,实现高性能、低延迟的设备驱动开发。
本节目的: 根据上节写的USB鼠标驱动,来依葫芦画瓢写出键盘驱动 1.首先我们通过上节的代码中修改,来打印下键盘驱动的数据到底是怎样的 先来回忆下,我们之前写的鼠标驱动的id_table是这样:...所以我们要修改id_table,使这个驱动为键盘的驱动,如下图所示: ? 然后修改中断函数,通过printk()打印数据: 我们先按下按键A为例,打印出0x04,如下图: ?...发现该数组的0X04就是0X30,看来要写个键盘驱动,还需要上面的数组才行..../kernel.h> #include linux/slab.h> #include linux/module.h> #include linux/init.h> #include linux/...5.4 使用exec 0</dev/tty1测试 (exec命令详解入口地址: http://www.cnblogs.com/lifexy/p/7553228.html) 如下图,就能通过板子上的键盘来操作了
本节便开始来写一个简单的虚拟网卡驱动,也就是说不需要硬件相关操作,所以就没有中断函数,我们通过linux的ping命令来实现发包,然后在发包函数中伪造一个收的ping包函数,实现能ping通任何ip地址...: #include linux/module.h> #include linux/errno.h> #include linux/netdevice.h> #include linux/etherdevice.h...> #include linux/kernel.h> #include linux/types.h> #include linux/fcntl.h> #include linux/interrupt.h...> #include linux/ioport.h> #include linux/in.h> #include linux/skbuff.h> #include linux/slab.h> #...include linux/spinlock.h> #include linux/string.h> #include linux/init.h> #include linux/bitops.h
https://blog.csdn.net/xuzhina/article/details/43969499 有的时候,开发时需要用到设备的pci ID,如用dpdk来绑定某个网卡,需要用网卡的...pci ID。...下面有一些方法是可以获取pci ID的。 1.使用lspci命令。...可以看到以太网卡的PCI ID是02:01.0 但有时候在一些嵌入式的系统里,lspci命令并不是由pciutils提供,而是由其它软件包提供,或者是修改过,输出结果可能如下: ?...Intel(R) PRO/1000 Network Connection 也可以看到以太网卡的PCI ID是02:01.0 这种方法无论是在通用的发行版或是定制的嵌入式系统,都适用。
hexdump: 查看文件的内容,比如二进制文件中包含的某些字符串,通常用来调试驱动用 描述: 我们以event1为例,当我们insmod挂载了键盘驱动后,出现一个event1设备, 此时没有按键按下,...通过键盘驱动的read函数,若有按键按下,就会上传按键数据给用户层hexdump 因为键盘驱动的input_handler 是:evdev_handler 所以键盘驱动的read函数是: evdev_handler...,有信息就会打印上面数据 1.调试键盘驱动 (键盘驱动代码:http://www.cnblogs.com/lifexy/p/7553861.html) 以按开发板的按键 KEY_L,为例(因为数据是从低到高打印的...,所以数据是反的): # hexdump /dev/event1 //按键键盘驱动 /*按下时:*/ //hexdump序列号 秒 微妙...(触摸屏驱动代码: ) /dev/event0 //触摸屏驱动 # hexdump /dev/event0 //hexdump序列号 秒
学了这么多驱动,不难推出DMA的编写套路: 1)注册DMA中断,分配缓冲区 2)注册字符设备,并提供文件操作集合fops -> 2.1)file_operations里设置DMA硬件相关操作,...来启动DMA 由于我们是用字符设备的测试方法测试的,而本例子只是用两个地址之间的拷贝来演示DMA的作用,所以采用字符设备方式编写 1.驱动编写之前,先来讲如何分配释放缓冲区、DMA相关寄存器介绍、使用...DMA3中断服务函数 //NULL:中断产生类型, 不需要,所以填NULL //1:表示中断时,传入中断函数的参数,本节不需要所以填1,切记不能填0,否则注册失败 2.接下来,我们便来写一个DMA的字符设备驱动...: #include linux/module.h> #include linux/kernel.h> #include linux/fs.h> #include linux/init.h> #...include linux/delay.h> #include linux/irq.h> #include #include <asm/arch/regs-gpio.h
1.无操作系统时的硬件、驱动、应用软件要满足高内聚、低耦合。 2.有操作系统时的驱动, 3.LINUX驱动与整个软硬件的关系
使得驱动代码,具有更好的扩展性和跨平台性,就不会因为新的平台而再次编写驱动 介绍: 分离就是在驱动层中使用platform机制把硬件相关的代码(固定的,如板子的网卡、中断地址)和驱动(会根据程序作变动...,通过这个总线将设备和驱动联系起来,属于Linux中bus的一种 该platform_bus_type的结构体定义如下所示(位于drivers/base): struct bus_type platform_bus_type.../module.h> #include linux/version.h> #include linux/init.h> #include linux/kernel.h> #include linux.../types.h> #include linux/interrupt.h> #include linux/list.h> #include linux/timer.h> #include linux.../init.h> #include linux/serial_core.h> #include linux/platform_device.h> static struct resource led_resource
领取专属 10元无门槛券
手把手带您无忧上云