Читаем Linux API. Исчерпывающее руководство полностью

При создании объекта разделяемой памяти пользователь и группа, которые им владеют, определяются на основе реальных пользовательского и группового идентификаторов процесса, вызвавшего shm_open(), а права доступа к объекту устанавливаются в соответствии со значением, указанным в битовой маске mode. Битовые значения аргумента mode аналогичны применяемым для файлов (см. табл. 15.4). Как и в случае с системным вызовом open(), к правам доступа, заданным в битовой маске, применяется атрибут umask (см. подраздел 15.4.6). Отличие от вызова open() состоит в том, что аргумент mode всегда является обязательным; если мы не создаем новый объект, то он должен быть равен 0.

Для файлового дескриптора, возвращаемого вызовом shm_open(), устанавливается флаг FD_CLOEXEC (см. раздел 27.4); благодаря этому он автоматически закрывается, когда процесс выполняет exec() (точно так же при выполнении exec() удаляются отображения).

Сразу после создания новый объект разделяемой памяти имеет нулевую длину. То есть перед вызовом mmap() обычно используется функция ftruncate(), которая устанавливает размер объекта (см. раздел 5.8). Ее можно вызвать и после mmap(), чтобы уменьшить или увеличить объект разделяемой памяти (при этом следует учитывать замечания, приведенные в подразделе 45.4.3).

При расширении объекта разделяемой памяти дополнительные байты изначально заполняются нулями.

В любой момент к файловому дескриптору, возвращенному функцией shm_open(), можно применить вызов fstat() (см. раздел 15.1). Данное действие позволит получить структуру stat, поля которой содержат информацию об объекте разделяемой памяти, включая его размер (st_size), права доступа (st_mode), владельца (st_uid) и группу (st_gid); это те поля, которые должны содержаться в структуре stat согласно стандарту SUSv3, однако, помимо них, Linux возвращает различные дополнительные сведения, включая данные о времени.

Информация о правах доступа и владельце объекта разделяемой памяти может быть изменена путем вызовов fchmod() и, соответственно, fchown().

Пример программы

В листинге 50.1 представлен простой пример использования вызовов shm_open(), ftruncate() и mmap(). Эта программа создает объект разделяемой памяти, размер которого задается с помощью аргумента командной строки, и отображает его на виртуальное адресное пространство процесса. (Этап отображения является избыточным, так как мы не собираемся ничего делать с разделяемой памятью; он здесь для демонстрации применения вызова mmap().) Программа позволяет задействовать аргументы командной строки, чтобы выбрать флаги (O_CREAT и O_EXCL) для функции shm_open().

В следующем примере мы используем данную программу для создания объекта разделяемой памяти размером 10 000 байт, а затем запускаем команду, чтобы показать этот объект в /dev/shm:

$ ./pshm_create — c /demo_shm 10000

$ ls — l /dev/shm

total 0

— rw- 1 mtk users 10000 Jun 20 11:31 demo_shm

Листинг 50.1. Создание объекта разделяемой памяти POSIX

pshm/pshm_create.c

#include

#include

#include

#include "tlpi_hdr.h"

static void

usageError(const char *progName)

{

fprintf(stderr, "Usage: %s [-cx] shm-name

size [octal-perms]\n", progName);

fprintf(stderr, " — c Create shared memory (O_CREAT)\n");

fprintf(stderr, " — x Create exclusively (O_EXCL)\n");

exit(EXIT_FAILURE);

}

int

main(int argc, char *argv[])

{

int flags, opt, fd;

mode_t perms;

size_t size;

void *addr;

flags = O_RDWR;

while ((opt = getopt(argc, argv, "cx"))!= -1) {

switch (opt) {

case 'c': flags |= O_CREAT; break;

case 'x': flags |= O_EXCL; break;

default: usageError(argv[0]);

}

}

if (optind + 1 >= argc)

usageError(argv[0]);

size = getLong(argv[optind + 1], GN_ANY_BASE, "size");

perms = (argc <= optind + 2)? (S_IRUSR | S_IWUSR):

getLong(argv[optind + 2], GN_BASE_8, "octal-perms");

/* Создаем объект разделяемой памяти и устанавливаем его размер */

fd = shm_open(argv[optind], flags, perms);

if (fd == -1)

errExit("shm_open");

if (ftruncate(fd, size) == -1)

errExit("ftruncate");

/* Отображаем объект разделяемой памяти */

addr = mmap(NULL, size, PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0);

if (addr == MAP_FAILED)

errExit("mmap");

exit(EXIT_SUCCESS);

}

pshm/pshm_create.c

50.3. Использование объектов разделяемой памяти

В листингах 50.2 и 50.3 демонстрируется применение объектов разделяемой памяти для передачи данных от одного процесса другому. Первая программа копирует строку в существующий объект разделяемой памяти; имя объекта указывается в первом аргументе командной строки, а копируемая строка — во втором. Но перед выполнением этих операций программа задействует вызов ftruncate(), чтобы размер объекта был равен размеру строки, которую нужно копировать.

Листинг 50.2. Копирование данных в объект разделяемой памяти POSIX

pshm/pshm_write.c

#include

#include

#include "tlpi_hdr.h"

int

main(int argc, char *argv[])

{

Перейти на страницу:

Похожие книги

1С: Бухгалтерия 8 с нуля
1С: Бухгалтерия 8 с нуля

Книга содержит полное описание приемов и методов работы с программой 1С:Бухгалтерия 8. Рассматривается автоматизация всех основных участков бухгалтерии: учет наличных и безналичных денежных средств, основных средств и НМА, прихода и расхода товарно-материальных ценностей, зарплаты, производства. Описано, как вводить исходные данные, заполнять справочники и каталоги, работать с первичными документами, проводить их по учету, формировать разнообразные отчеты, выводить данные на печать, настраивать программу и использовать ее сервисные функции. Каждый урок содержит подробное описание рассматриваемой темы с детальным разбором и иллюстрированием всех этапов.Для широкого круга пользователей.

Алексей Анатольевич Гладкий

Программирование, программы, базы данных / Программное обеспечение / Бухучет и аудит / Финансы и бизнес / Книги по IT / Словари и Энциклопедии
1С: Управление торговлей 8.2
1С: Управление торговлей 8.2

Современные торговые предприятия предлагают своим клиентам широчайший ассортимент товаров, который исчисляется тысячами и десятками тысяч наименований. Причем многие позиции могут реализовываться на разных условиях: предоплата, отсрочка платежи, скидка, наценка, объем партии, и т.д. Клиенты зачастую делятся на категории – VIP-клиент, обычный клиент, постоянный клиент, мелкооптовый клиент, и т.д. Товарные позиции могут комплектоваться и разукомплектовываться, многие товары подлежат обязательной сертификации и гигиеническим исследованиям, некондиционные позиции необходимо списывать, на складах периодически должна проводиться инвентаризация, каждая компания должна иметь свою маркетинговую политику и т.д., вообщем – современное торговое предприятие представляет живой организм, находящийся в постоянном движении.Очевидно, что вся эта кипучая деятельность требует автоматизации. Для решения этой задачи существуют специальные программные средства, и в этой книге мы познакомим вам с самым популярным продуктом, предназначенным для автоматизации деятельности торгового предприятия – «1С Управление торговлей», которое реализовано на новейшей технологической платформе версии 1С 8.2.

Алексей Анатольевич Гладкий

Финансы / Программирование, программы, базы данных