После того как файл был отображен в памяти, изменить права доступа к нему позволяет функция mprotect()
Учтите, что память, выделяемая функцией malloc()
malloc(), нужно запросить более крупный блок, а затем найти в нем участок, выровненный по границе страниц.Кроме того, с помощью функции mmap()
malloc() и запрашивать память непосредственно у ядра Linux.Предположим, к примеру, что программа выделяет страницу, отображая в памяти файл /dev/zero
int fd = open("/dev/zero", O_RDONLY);
char* memory =
mmap(NULL, page_size, PROT_READ | PROT_WRITE,
MAP_PRIVATE, fd, 0);
close(fd);
Далее программа запрещает запись в эту область памяти, вызывая функцию mprotect()
mprotect(memory, page_size, PROT_READ);
Существует оригинальная методика контроля памяти: можно защитить область памяти с помощью функций mmap()
mprotect(), а затем обрабатывать сигнал SIGSEGV, посылаемый при попытке доступа к этой памяти. Эта методика иллюстрируется в листинге 8.7.mprotect()#include
#include
#include
#include
#include
#include
#include
#include
static int alloc_size;
static char* memory;
void segv_handler(int signal_number) {
printf("memory accessed!\n");
mprotect(memory, alloc_size, PROT_READ | PROT_WRITE);
}
int main() {
int fd;
struct sigaction sa;
/* Назначение функции segv_handler() обработчиком сигнала
SIGSEGV. */
memset(&sa, 0, sizeof(sa));
sa.sa_handler = &segv_handler;
sigaction(SIGSEGV, &sa, NULL);
/* Выделение одной страницы путем отображения в памяти файла
/dev/zero. Сначала память доступна только для записи. */
alloc_size = getpagesize();
fd = open("/dev/zero", O_RDONLY);
memory =
mmap(NULL, alloc_size, PROT_WRITE, MAP_PRIVATE, fd, 0);
close(fd);
/* Запись на страницу для получения ее копии в частное
использование. */
memory[0] = 0;
/* Запрет на запись в память. */
mprotect(memory, alloc_size, PROT_NONE);
/* Попытка записи в память. */
memory[0] = 1;
/* Удаление памяти. */
printf("all done\n");
munmap(memory, alloc_size);
return 0;
}
Программа работает по следующей схеме.
1. Задается обработчик сигнала SIGSEGV
2. Файл /dev/zero
3. Программа защищает память, вызывая функцию mprotect()
PROT_NONE.4. Когда программа впоследствии обращается к памяти, Linux посылает ей сигнал SIGSEGV
segv_handler(). Обработчик сигнала отменяет защиту памяти, разрешая выполнить операцию записи.5. Программа удаляет область память с помощью функции munmap()
8.10. Функция nanosleep(): высокоточная пауза