boot和内核中寄存器操作的区别

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中一致了。感谢博主,转载一下表示支持。

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

lyx_wmt

能白嫖就绝不打赏

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值