/* When trying to map /dev/mem with offset 0xFFFFF000 on the ARM platform, mmap * returns -EOVERFLOW. * * Since off_t is defined as a long int and the sign bit is set in the address, * the shift operation shifts in ones instead of zeroes * from the left. This results the offset sent to the kernel function becomes * 0xFFFFFFFF instead of 0x000FFFFF with MMAP2_PAGE_SHIFT set to 12. */ #include #include #include #include #include #include #include #define FATAL do { fprintf(stderr, "Error at line %d, file %s (%d) [%s]\n", \ __LINE__, __FILE__, errno, strerror(errno)); exit(1); } while(0) #define MAP_SIZE sysconf(_SC_PAGESIZE) #define MAP_MASK (MAP_SIZE - 1) int main(int argc, char **argv) { void* map_base = 0; int fd; off_t target = 0xfffff000; if((fd = open("/dev/mem", O_RDWR | O_SYNC)) == -1) { /* skip test for non-root users */ if (errno == EACCES) return 0; FATAL; } printf("/dev/mem opened.\n"); fflush(stdout); /* Map one page */ map_base = mmap(0, MAP_SIZE, PROT_READ | PROT_WRITE, MAP_SHARED, fd, target & ~MAP_MASK); if(map_base == (void *) -1) FATAL; printf("Memory mapped at address %p.\n", map_base); fflush(stdout); if(munmap(map_base, MAP_SIZE) == -1) FATAL; close(fd); return 0; }