Рейтинговые книги
Читем онлайн Linux: Полное руководство - Денис Колисниченко

Шрифт:

-
+

Интервал:

-
+

Закладка:

Сделать
1 ... 139 140 141 142 143 144 145 146 147 ... 158

 unsigned short shm_npages;

 /* размеры сегмента (в страницах) */

 /* массив указателей на $frames -> S$*/

 unsigned long *shm_pages;

 struct vm_area_struct *attaches;

 /* дескрипторы для привязок */

};

Я немного сократил эту структуру, оставив описание только нужных нам полей. Полагаю, что вы в нем разберетесь. Возможно, вас заинтересовали термины «привязка» и «отвязка». Привязка — это размещение сегмента в адресном пространстве процесса, подключение к разделяемому сегменту памяти (РСП). Отвязка, соответственно, — отключение, Поле shm_nattch содержит количество привязок к РСП на данный момент.

Для создания нового РСП используется системный вызов shmget(). Этот же вызов используется для подключения к уже существующему РСП.

int shmget(key_t key, int size, int shmflg);

Первый аргумент — это ключ IPC, полученный с помощью ftok(), второй — размер РСП в байтах, а третий — флаги системного вызова shmget. Если установлен флаг IPC_CREAT, системный вызов создаст новый РСП или подключится к уже существующему сегменту, если обнаружится, что уже есть такой сегмент (с таким же значением ключа). Если установлен флаг IPC_EXCL вместе с IPC_CREAT (сам по себе он бесполезен) подключение к существующему РСП запрещается.

Системный вызов shmget() возвращает идентификатор РСП или -1, если произошла ошибка. Переменная errno устанавливается так:

♦ EACCESS — не хватает полномочий для доступа к сегменту;

♦ EINVAL — неправильно заданы размеры сегмента;

♦ EEXISTS — сегмент уже существует, создание невозможно. Вы получите эту ошибку, если будете использовать флаг IPC_EXCL вместе с IPC_CREAT при условии, что сегмент уже существует;

♦ IDRM — сегмент помечен на удаление или уже удален;

♦ ENOMEM — не хватает памяти для создания сегмента.

Приведем пример функции открытия/создания РСП:

int open_shms(key_t key, int size) {

 return (shmget(key, size, IPC_CREAT | 0660 )) == -1));

}

После получения идентификатора РСП мы должны «привязаться» к этому сегменту, то есть разместить сегмент в своем адресном пространстве. Для этого используется системный вызов shmat() (shared memory attachment):

int shmat(int shmid, char *shmaddr, int shmflg);

Первый аргумент — это идентификатор РСП, который мы получаем с помощью предыдущего вызова. Второй аргумент — это адрес привязки. Если указать вместо адреса ноль, то ядро само найдет нераспределенную область.

Третий аргумент — это флаги. Обычно используется два флага:

♦ SHM_RND — переданный адрес будет округлен до ближайшей страницы (если вы сами указываете адрес);

♦ SHM_RDONLY — РСП будет доступен только для чтения.

В случае успеха shmat() возвращает адрес, по которому сегмент был привязан к процессу, или -1, если произошла ошибка. Переменная errno может принимать всего три значения:

♦ EACCESS — нет доступа;

♦ ENOMEM — не хватает памяти;

♦ EINVAL — ошибка в параметрах, то есть неправильное значение ID или адреса привязки (shmaddr).

Пример привязки:

char *ptr;

prt = shmat(sh_id,0,0);

После привязки сегмента к адресному пространству доступны операции чтения и записи, которые очень напоминают работу с простыми указателями.

Для снятия привязки используется системный вызов shmdt():

int shmdt(char *shmaddr);

В случае ошибки данный системный вызов возвращает -1. Значение errno только одно: EINVAL, то есть вы неправильно указали адрес привязки. После отвязки значение элемента shm_nattch структуры shmid_ds уменьшается на 1. Если больше нет привязок, то есть shm_nattch = 0, сегмент будет удален ядром.

Для управления РСП используется системный вызов shmctl():

int shmctl(int shmid, int cmd, struct shmid_ds *buf);

Первый аргумент — это идентификатор РСП, второй — команда, а третий — буфер для команд IPC_STAT/IPC_SET. Команд для управления три:

♦ IPC_STAT — сохраняет структуру shmid_ds по адресу buf;

♦ IPC_SET — берет значение элемента ipc_perm структуры shmid_ds и устанавливает его для сегмента. Значение берется из buf;

♦ IPC_RMID — помечает сегмент для удаления, само удаление произойдет, как только последний процесс отвяжется от сегмента. Если сегмент помечен на удаление, ни один процесс не сможет привязаться к сегменту.

В случае успеха системный вызов shmctl() возвращает 0 или -1, если произошла ошибка. Переменная errno устанавливается так:

♦ EACCESS — нет прав;

♦ EFAULT — ошибочный адрес buf;

♦ EIDRM — сегмент помечен на удаление;

♦ EINVAL — неправильный идентификатор сегмента.

Вот теперь мы готовы к написанию демонстрационной программы для работы с разделяемыми сегментами памяти.

Листинг 26.6. Демонстрационная программа shm_demo.c

#include <stdio.h>

#include <sys/types.h>

#include <sys/ipc.h>

#include <sys/shm.h>

/* размер нашего сегмента - 256 байтов */

#define SIZE 256

int main(int argc, char *argv[]) {

 key_t key; /* ключ */

 int shmid, с; /* идентификатор */

 char *ptr; /* указатель, через который мы будем

               работать с сегментом */

 /* Если аргументы не указаны ... */

 if (argc == 1) {

  printf("shm_demo usage:n");

  printf("shm_demo -w string записать строку в сегментn");

  printf("shm_demo -r прочитать инф. из сегментаn");

  printf("shm_demo -d удалить сегментn");

  printf("shm_demo -m mode изменить права доступаn");

  exit(1);

 }

 /* Генерируем ключ IPC */

 key = ftok(".", 'D');

 /* Пытаемся создать сегмент */

 if ((shmid =

  shmget(key, SIZE, IPC_CREAT|IPC_EXCL|0660)) == -1) {

  printf("Сегмент существует, подключаемся к нему...n");

  /* Используем shmget без IPC_EXCL */

  if ((shmid = shmget(key, SIZE, 0)) == -1) {

   printf("Ошибка в shmgetn");

   exit(1);

  }

 } else {

  printf("Создаем новый сегментn");

 }

 /* Привязываемся к сегменту */

 if ((ptr = shmat(shmid, 0, 0)) == -1) {

  perror("shmat");

  exit(1);

 }

 /* Разбираем параметры командной строки:

  w - запись в сегмент

  r - чтение

  d - удаление сегмента

  m — изменение прав доступа */

 switch(tolower(argv[1][1])) {

 case 'w':

  shm_write(shmid, ptr, argv[2]);

  break;

 case 'r':

  shm_read(shmid, ptr);

  break;

 case 'd':

  shm_rm(shmid);

  break;

 case 'm':

  shm_change_mode(shmid, argv[2]);

  break;

 }

}

/* Функция для записи в сегмент: ей нужно передать

ID сегмента, адрес привязки и записываемую информацию */

shm_write(int shmid, char *ptr, char *info) {

 strcpy(ptr, info);

}

/* Функция чтения информации из сегмента */

shm_read(int shmid, char *ptr) {

 printf("Информация из сегмента: %sn", ptr);

}

/* Функция удаления сегмента */

shm_rm(int shmid) {

 shmctl(shmid, IPC_RMID, 0);

 printf("Сегмент помечен на удалениеn");

}

/* Функция изменения прав доступа. Ей нужно передать

 идентификатор сегмента и права доступа в виде строки,

 например, "0666" * /

shm_change_mode(int shmid, char *mode) {

 struct shmid_ds mds;

 shmctl(shmid, IPC_STAT, &mds);

 printf("Старые права доступа: %on", mds.shm_perm.mode);

 sscanf(mode, "%o", &mds.shm_perm.mode);

 shmctl(shmid, IPC_SET, &mds);

 printf("Новые права доступа: %on", mds.shm_perm.mode);

}

Использовать программу нужно так:

./shm_demo -w строка

запись строки в сегмент

./shm_demo -r

чтение строки из сегмента

./shmdemo -m права

изменение прав доступа

./shm_demo -d

удаление сегмента

Выполните команду

$ ./shm_demo -w string

А затем запустите утилиту ipcs. Вы увидите, что наша программа создала разделяемый сегмент памяти:

------ Shared Memory Segments ------

key shmid owner perms bytes nattch status

0x44063781 0 root 660 256 0

------ Semaphore Arrays ------

key semid owner perns nsems status

------ Message Queues ------

key msqid owner perms used-bytes messages

Затем выполните команду:

$ ./shm_demo -r

Вы получите информацию:

Информация из сегмента: string

Попробуем изменить права доступа, а затем просмотреть информацию командой ipcs:

$./shm_demo -m 0666

------ Shared Memory Segments ------

key        shmid owner perms bytes nattch status

0x44063781     0 root    660   256        0

------ Semaphore Arrays ------

1 ... 139 140 141 142 143 144 145 146 147 ... 158
На этой странице вы можете бесплатно читать книгу Linux: Полное руководство - Денис Колисниченко бесплатно.
Похожие на Linux: Полное руководство - Денис Колисниченко книги

Оставить комментарий