uboot下和内核下操作cpu寄存器的区别
文件为转载基础上加上自己遇到的具体现象,转载地址为:
https://blog.csdn.net/csdnxmj/article/details/105753733
#1.在uboot中操作CPU寄存器
#include <asm/io.h>
reg = readl(PHY_ADDR);
reg &= ~IOMUXC_REG_GPR1_ACTCS0_MASK;
writel(reg, PHY_ADDR);
其中PHY_ADDR是物理地址,跟踪代码发现writel操作如下:
#define writel(v,a) __arch_putl(v,a)
#define __arch_putl(v,a) (*(volatile unsigned int *)(a) = (v))
所以在uboot里面配置寄存相当于是给物理地址直接赋值,volatile的意思是提醒编译器需要存储或读取这个变量的时候,都会直接从变量地址中读取数据
#2.在kernel中操作CPU寄存器
#include <asm/io.h>
而在内核中,上面的写法是无法运行的,会提示虚拟地址错误。在内核中通常是通过虚拟地址来给物理地址赋值,所以需要将物理地址转换成虚拟地址
reg = readl(ioremap(PHY_ADDR,4));
reg &= ~IOMUXC_REG_GPR1_ACTCS0_MASK; //reg operation
reg |= ((CS0_NORFLASH_SIZE | IOMUXC_REG_GPR1_ACTCS0));
writel(reg, ioremap(PHY_ADDR,4));
这里的ioremap是将物理地址PHY_ADDR转换得到对应的虚拟地址,4表示4个字节,即32位的地址
#3.总结
在uboot & kernel中都可以使用readl()和writel()两个函数来对CPU的寄存器进行操作,但是区别在于,在uboot中进行读写的是CPU上实际物理地址,而在kernel中读写的是将物理地址经过内存映射后的虚拟地址。
#4.真实情况
因为开发问题,没找到寄存器手册,就从boot中找到了两个gpio的初始化地址,直接在内核中readl该地址,结果直接死机了。搜到了这篇文章,再去内核中读该寄存器地址,就和boot中一致了。感谢博主,转载一下表示支持。