我在飞思卡尔iMX6 arm处理器上运行3.14.14内核的debian。
我在一个嵌入式应用程序中有这段代码,它给出了一个分段错误。
volatile unsigned int& GPIO2IO26CTRL = *((volatile unsigned int*)0x20e0104);
GPIO2IO26CTRL = 0x5;
还尝试了:
volatile unsigned int* GPIO2IO26CTRL = (volatile unsigned int*)0x20e0104;
*GPIO2IO26CTRL = 0x5;
但是我真的想把0x5写到位置0x20e0104。这告诉iMX6芯片Io线将是GPIO而不是spi;
我该如何解决这个问题呢?
发布于 2015-05-21 02:13:24
这在用户空间是可能的(尽管您可能需要是root用户)。关键是将内存重新映射到进程的地址空间。
有一个名为devmem2
的程序可以做到这一点。源代码是here。
发布于 2015-05-22 03:44:59
感谢大家的帮助。如果有人有同样的问题,我想展示我的解决方案。这些代码大部分来自Jan-Derk Bakker的devmem2。
下面是我编写的两个从内存中读取和写入的例程。
static const u_int32_t MAP_SIZE = 4096;
static const u_int32_t MAP_MASK = (MAP_SIZE - 1);
bool CGpio::writeMem(u_int32_t memAddr, u_int32_t &value)
{
int fd;
void *map_base, *virt_addr;
off_t target = memAddr;
// open the mem file
if((fd = open("/dev/mem", O_RDWR | O_SYNC)) == -1)
return false;
// map one page
map_base = mmap(0, MAP_SIZE, PROT_READ | PROT_WRITE, MAP_SHARED, fd, target & ~MAP_MASK);
if (map_base == (void *)-1)
return false;
// map virutial space
virt_addr = map_base + (target & MAP_MASK);
// write the value
*((u_int32_t *) virt_addr) = value;
// clean up
bool ret = (munmap(map_base, MAP_SIZE) != -1);
close(fd);
return ret;
}
bool CGpio::readMem(u_int32_t memAddr, u_int32_t &value)
{
int fd;
void *map_base, *virt_addr;
off_t target = memAddr;
// open the mem file
if((fd = open("/dev/mem", O_RDWR | O_SYNC)) == -1)
return false;
// map one page
map_base = mmap(0, MAP_SIZE, PROT_READ | PROT_WRITE, MAP_SHARED, fd, target & ~MAP_MASK);
if (map_base == (void *)-1)
return false;
// map virutial space
virt_addr = map_base + (target & MAP_MASK);
// read the value
value = *((u_int32_t *) virt_addr);
// clean up
bool ret = (munmap(map_base, MAP_SIZE) != -1);
close(fd);
return ret;
}
下面是我使用的头文件。
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h>
#include <signal.h>
#include <fcntl.h>
#include <sys/mman.h>
#include <string.h>
#include <iostream>
#include <sys/types.h>
https://stackoverflow.com/questions/30354405
复制相似问题