举个典型的例子吧,首先我不建议你在bar0空间里放数据,太浪费,bar0空间就放一些控制用的寄存器吧,相对数据而言,没多少的。
对FPGA地址的读写,假设FPG**内你设计了一个地址查找表,某寄存器byte地址为0x000055aa,那么你在PC端往bar0基址+0x000055aa写一个byte数0x1,那么这个寄存器的值就是0x1。
对大段的数据而言,你在内存中开辟一段存储空间,比如0xcc000000~0xdd000000,然后在FPGA里头做一个DMA控制器,通过PC往FPGA的寄存器里写DMA起始地址0xcc000000和dma的长度**x,然后你做的这个DMA控制器开始向PCIE IP发送读请求,读地址是0xcc000000+不停累加的偏址,当读到指定长度后,DMA控制器停止工作,并且发一个中断给PC。
上面这个是常用的方法
对FPGA地址的读写,假设FPG**内你设计了一个地址查找表,某寄存器byte地址为0x000055aa,那么你在PC端往bar0基址+0x000055aa写一个byte数0x1,那么这个寄存器的值就是0x1。
对大段的数据而言,你在内存中开辟一段存储空间,比如0xcc000000~0xdd000000,然后在FPGA里头做一个DMA控制器,通过PC往FPGA的寄存器里写DMA起始地址0xcc000000和dma的长度**x,然后你做的这个DMA控制器开始向PCIE IP发送读请求,读地址是0xcc000000+不停累加的偏址,当读到指定长度后,DMA控制器停止工作,并且发一个中断给PC。
上面这个是常用的方法
在单个BAR里头,处理器分配的内存实际上是到根节点存储空间的映射。对PC系统来说就是桥片了。以Intel的南北桥结构为例,操作系统将地址,数据打包以后,经过北桥的内存控制器,通过DMA总线发到南桥上。南桥上的root端(根据经验猜测,有一堆128Byte深度缓存),将每个包的数据放入一个独立的缓存中,然后根据这些包的VC号,事务号,以及对应通道的信用值来排列发送顺序。同时向端点汇报自己的信用值(缓存剩余量)。这里对操作系统内存而言,应该是FIFO式的,但是对root而言,应该是类似双口的结构。
我没写过驱动,具体请freefpga指正
我没写过驱动,具体请freefpga指正
回复
有奖活动 | |
---|---|
【有奖活动——B站互动赢积分】活动开启啦! | |
【有奖活动】分享技术经验,兑换京东卡 | |
话不多说,快进群! | |
请大声喊出:我要开发板! | |
【有奖活动】EEPW网站征稿正在进行时,欢迎踊跃投稿啦 | |
奖!发布技术笔记,技术评测贴换取您心仪的礼品 | |
打赏了!打赏了!打赏了! |