そのため、私はカーネル ドライバーに関してまったくの初心者であり、ioremap 関数について質問があります。
ARM Cortex-M3 と FPGA ファブリックを搭載した SoC 上のカスタム VHDL モジュールで定義されたいくつかのレジスタにアクセスするためのドライバーを作成しています。例を見て、ioremap を使用する必要があると考えましたが、Cortex-M3 には MMU がないため、次の例のようにポイントがわかりません。
/* Physical addresses */
static u32* rcu_trig_recv_physaddr = ((u32 *) 0x50040000);
static int rcu_trig_recv_size = 0x10; // size of 16 for testing
/* Virtual addresses */
static u32* rcu_trig_recv_virtbase = NULL;
/*removed code not relevant for the question*/
static int __init rcumodule_init(void)
{
int iResult = 0; // holding result of operations
u32 buffer;
// Register the driver
iResult = register_chrdev(rcuc_majorID, "rcuc", &rcuc_fops);
if (iResult < 0) {
printk(KERN_INFO "module init: can't register driver\n");
}
else{
printk(KERN_INFO "module init: success!\n");
}
// Map physical address to virtual address
if(rcu_trig_recv_size){
rcu_trig_recv_virtbase = (u32*) ioremap_nocache( (u32 *)rcu_trig_recv_physaddr, rcu_trig_recv_size );
printk("Remapped TRGRECV from 0x%p to 0x%p\n", rcu_trig_recv_physaddr, rcu_trig_recv_virtbase);
}
// try to read some stuff, expecting 0x17240f09
buffer = readl(rcu_trig_recv_virtbase);
printk("read %lx, at 0x%p\n", buffer, rcu_trig_recv_virtbase);
return iResult;
}
ドライバーを insmod すると、これが返されます。
# insmod trigger.ko
module init: success!
Remapped TRGRECV from 0x50040000 to 0x50040000
read 17240f09, at 0x50040000
これによると、代わりに物理アドレスを読み取ったほうがよいでしょう。それとも、それは悪い考えであり、より良い方法でレジスタをいじる必要がありますか?