Can';t读写(内存映射)硬件寄存器

Can';t读写(内存映射)硬件寄存器,c,memory,embedded-linux,C,Memory,Embedded Linux,我正在开发LPC3131(NXP的arm9),我正在尝试在linux内核中制作我的第一个驱动程序,但是我遇到了一些麻烦 事实上,我想读/写微控制器寄存器中的值,就像我在那里发现的一样 我在驱动程序的init函数中执行以下操作 static int parr_init(void) { int result; dev_t my_dev; my_cdev = cdev_alloc(); cdev_init(my_cdev, &p

我正在开发LPC3131(NXP的arm9),我正在尝试在linux内核中制作我的第一个驱动程序,但是我遇到了一些麻烦

事实上,我想读/写微控制器寄存器中的值,就像我在那里发现的一样

我在驱动程序的init函数中执行以下操作

static int parr_init(void)
{
        int result;
        dev_t my_dev;
        my_cdev = cdev_alloc();
        cdev_init(my_cdev, &parr_port_fops);

        printk(KERN_ALERT "parr_driver : Hello world \n");

        result = alloc_chrdev_region(&my_dev, 0, 0, "parr_driver");

        if(result < 0)
        {
                printk(KERN_WARNING "parr_driver : can't get a major number \n");
        }
        else
        {
                printk(KERN_ALERT "parr_driver : major number : %i \n", MAJOR(my_dev));
                printk(KERN_ALERT "parr_driver : minor number : %i \n", MINOR(my_dev));
        }

        result = cdev_add(my_cdev, my_dev, 1);

        if(result < 0)
        {
                printk(KERN_ALERT "parr_driver : failed to register driver \n");
        }
        else
        {
                printk(KERN_ALERT "parr_driver : driver is now ready to be use \n");
        }
                                uint32_t volatile* reg = (uint32_t volatile *) 0x13009000;

                                printk("reg adress %p \n", reg);

                                printk("reg value  %X \n", *reg);
        return 0;
}
控制台日志

parr_driver : minor number : 0 
parr_driver : driver is now ready to be use 
reg value  DFB 
reg value  DFB 
现在它不再崩溃了,读取似乎可以工作,但我仍然无法写入我的硬件寄存器(第二次是DFB而不是FFFF)


有什么想法吗?

正如克里斯·斯特拉顿(Chris Stratton)所描述的,这是一个访问的问题。实际上,在内核引导时,寄存器时钟是禁用的,这就是为什么我不能编写它


非常感谢您的帮助。

1)为什么不使用uint16\U t*?2) 该平台是否有一个“平面”内存映射,或者是否涉及mmu(和虚拟寻址)?在我看来很像mmu。需要ioremap()吗?我使用uint32_t*而不是uint16_t,因为它是一个32位的µC。实际上有一个mmu,所以我必须使用ioremap,现在我将测试它。我还将查看处理器数据表,找出访问给定内存映射寄存器的“规则”。有时,给定的访问宽度是必需的。有时你还需要在其他地方写入一个位来解锁寄存器,或者甚至在使用它之前启用芯片所在部分的电源。@ChrisStratton-embedded fun:)今年最糟糕的情况是——三天之后发现我在PCON寄存器中设置了错误的位,can控制器没有电源:(祝贺你解决了这个问题!对不起,我没有明确提到打卡——既然你提出了这个问题,我记得我自己也遇到了这个问题。我知道这样做感觉很愚蠢,但你可以接受你自己的答案,这样问题就解决了。
    if(request_mem_region(0x13009000, 4, "parr_driver") != NULL)
    {
            map = ioremap(0x13009000, 4);

            if(map != NULL)
            {
                    printk("reg value  %X \n", ioread32(map));

                    iowrite32(0x0000FFFF, map);

                    printk("reg value  %X \n", ioread32(map));
            }
            else
            {
                    printk("ioremap FAIL \n");
            }
    }
parr_driver : minor number : 0 
parr_driver : driver is now ready to be use 
reg value  DFB 
reg value  DFB