Sto cercando di usare mmap nello spazio utente per leggere la memoria fisica in cui inizia "mem_map". È un array che contiene tutte le pagine fisiche. Questa è una macchina i386 con kernel 3.0 in esecuzione.mmap: Operazione non consentita
il codice è simile a questo:
....
//define page size
//
#define PAGE_SIZE 0x1000 //4096 bytes
#define PAGE_MASK (PAGE_SIZE - 1)
....
/* open /dev/mem file*/
if((fd = open("/dev/mem", O_RDWR | O_SYNC)) == -1) {
printf("/dev/mem could not be opened.\n");
perror("open");
exit(1);
} else {
printf("/dev/mem opened.\n");
}
/* Map one page */
printf(" mem_map is at physical addr: 0x%x\n", mem_map_phy_addr);
map_base = mmap(0, PAGE_SIZE, PROT_READ, MAP_SHARED, fd, (mem_map_phy_addr & ~PAGE_MASK)); //mem_map_phy_addr is at 0x356f2000
if(map_base == (void *) -1) {
printf("Memory map failed. err num = %d\n",errno);
perror("mmap"); //failed here
} else {
printf("Memory mapped at address %p.\n", map_base);
}
mi sono imbattuto questo come una radice. L'output è:
/dev/mem opened.
mem_map is at physical addr: 0x356f2000
Memory map failed. err num = 1
mmap: Operation not permitted
A dire il vero, ho cercato con Google il problema e ha aggiunto la seguente riga al mio file /etc/sysctl.conf:
vm.mmap_min_addr = 0
Ma questo non funziona neanche.
Qualcuno sa perché un'operazione mem_map come questa non è consentita e come posso aggirarla?
Grazie.
FYI, non è corretto utilizzare 'x & ~ PAGE_MASK'. Sui sistemi a 64 bit, questo troncerà gli indirizzi a 32 bit. Devi lanciare su 'uintptr_t' o un tipo di larghezza equivalente prima di prendere il complemento. –
Hai eseguito il comando sysctl per impostare il valore per mmap_min_addr o semplicemente modificare il file conf? Devi fare entrambe le cose. –
sì, ho fatto "sysctl -p" in seguito. – user899159