| /* 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 <unistd.h> | 
 | #include <stdio.h> | 
 | #include <stdlib.h> | 
 | #include <string.h> | 
 | #include <errno.h> | 
 | #include <fcntl.h> | 
 | #include <sys/mman.h> | 
 |  | 
 | #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 4096UL | 
 | #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; | 
 | } |