我正在使用C中的一个内核模块与PCIe卡对话,我已经使用pci_iomap分配了一些io内存,我使用ioread/write 32在那里写/读。
这是可行的,但性能非常差,我读到我可以通过memcpy_toio/fromio使用块传输,而不是一次只执行32b。
要写,我正在使用iowrite32(buffer[i], privdata->registers + i);
要阅读,我做buffer[i] = ioread32(&privdata->registers[i]);
我试图用下面的for循环替换这些循环:
memcpy_toio(privdata->registers, buffer, 2048);
memcpy_fromio(buffer, privdata->registers, 2048);如果我只用memcpy_toio替换写循环,并使用ioread32进行读取,程序不会崩溃,但指令似乎不会做任何事情(寄存器不会改变);
另外,当我用memcpy_fromio指令替换read循环时,它会崩溃。
我想这可能是因为读取器试图访问mem位置,而它还在被写入。有没有一种方法可以在iowrite32或memcpy_toio之后刷新写入队列?
我在这里做错什么了?
发布于 2015-02-18 15:02:46
只有当I/O内存行为类似于内存时才能使用memcpy_from/toio(),也就是说,如果可以推测地读取值,并且可以多次写入或无序写入。
标记为非预取的I/O范围不支持这一点。
发布于 2022-06-14 09:59:07
我不知道我的建议是否有效,单个ioread32函数比memcpy更有效,读取函数只需要读取或写入PCI设备一次,而memcpy则需要多次读写。内核提供了诸如ioread32_rep这样的函数来替换您为了提高效率而需要的繁琐的.If(本质上是相同的),您可以尝试使用ioread32_rep,也可以尝试使用memcpy来读取和写入可变长度的内容。
发布于 2015-02-16 10:15:48
您使用哪种缓冲区类型?
static inline void
memcpy_fromio(void *dst, volatile void __iomem *src, int count)
{
memcpy(dst, (void __force *) src, count);
}
static inline void
memcpy_toio(volatile void __iomem *dst, const void *src, int count)
{
memcpy((void __force *) dst, src, count);
}你可以看到简单的备忘录电话。
看看iowrite32()和ioread32()实现:
static inline void iowrite32(u32 val, void __iomem *p)
{
if (__is_PCI_addr(p))
val = _swapl(val);
__builtin_write32(p, val);
if (__is_PCI_MEM(p))
__flush_PCI_writes();
}
static inline unsigned int ioread32(void __iomem *p)
{
uint32_t ret = __builtin_read32(p);
if (__is_PCI_addr(p))
ret = _swapl(ret);
return ret;
}https://stackoverflow.com/questions/28518336
复制相似问题