1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
|
/* 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 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;
}
|