Марк Митчелл - Программирование для Linux. Профессиональный подход
- Название:Программирование для Linux. Профессиональный подход
- Автор:
- Жанр:
- Издательство:Вильямс
- Год:2002
- Город:Москва
- ISBN:5-8459-0243-6
- Рейтинг:
- Избранное:Добавить в избранное
-
Отзывы:
-
Ваша оценка:
Марк Митчелл - Программирование для Linux. Профессиональный подход краткое содержание
Данная книга в основном посвящена программированию в среде GNU/Linux. Авторы применяют обучающий подход, последовательно излагая самые важные концепции и методики использования расширенных возможностей системы GNU/Linux в прикладных программах. Читатели научатся писать программы, к интерфейсу которых привыкли пользователи Linux; освоят такие технологии, как многозадачность, многопотоковое программирование, межзадачное взаимодействие и взаимодействие с аппаратными устройствами; смогут улучшить свои программы, сделав их быстрее, надежнее и безопаснее; поймут особенности системы GNU/Linux, ее ограничения, дополнительные возможности и специфические соглашения.
Книга предназначена для программистов, уже знакомых с языком С и имеющих базовый опыт работы в GNU/Linux.
Программирование для Linux. Профессиональный подход - читать онлайн бесплатно полную версию (весь текст целиком)
Интервал:
Закладка:
#include
#include
int main() {
int segment_id;
char* shared_memory;
struct shmid_ds shmbuffer;
int segment_size;
const int shared_segment_size = 0x6400;
/* Выделение совместно используемого сегмента. */
segment_id =
shmget(IPC_PRIVATE, shared_segment_size,
IPC_CREAT | IPC_EXCL | S_IRUSR | S_IWUSR);
/* Подключение сегмента. */
shared_memory = (char*)shmat(segment_id, 0, 0);
printf("shared memory attached at address %p\n",
shared_memory);
/* Определение размера сегмента. */
shmctl(segment_id, IPC_STAT, &shmbuffer);
segment_size = shmbuffer.shm_segsz;
printf("segment size: %d\n", segment_size);
/* Запись строки в сегмент. */
sprintf(shared_memory, "Hello, world.");
/* Отключение сегмента. */
shmdt(shared_memory);
/* Повторное подключение сегмента, но по другому адресу! */
shared_memory =
(char*)shmat(segment_id, (void*) 0x5000000, 0);
printf("shared memory reattached at address %p\n",
shared_memory);
/* Отображение строки, хранящейся в совместно используемой
памяти. */
printf("%s\n", shared_memory);
/* Отключение сегмента. */
shmdt(shared_memory);
/* Освобождение сегмента. */
shmctl(segment_id, IPC_RMID, 0);
return 0;
}
5.1.7. Отладка
Команда ipcs
выдает информацию о взаимодействии процессов, включая сведения о совместно используемых сегментах (для этого следует задать флаг -m
). Например, в показанном ниже случае сообщается о том, что используется один такой сегмент, с номером 1627649:
% ipcs -m
-------- Shared Memory Segments --------
key shmid owner perms bytes nattch status
0x00000000 1627649 user 640 25600 0
Если этот сегмент был по ошибке "забыт" какой-то программой, его можно удалить с помощью команды ipcrm
:
% ipcrm shm 1627649
5.1.8. Проблема выбора
Благодаря совместному использованию памяти можно организовать быстрое двустороннее взаимодействие произвольного числа процессов. Любой пользователь сможет получать доступ к сегментам памяти для чтения/записи, но для этого программа должна следовать определенным правилам, позволяющим избегать конкуренции (чтобы, например, информация не оказалась перезаписанной до того, как будет прочитана). К сожалению, Linux не гарантирует монопольный доступ к сегменту, даже если он был создан с указанием флага IPC_PRIVATE
.
Кроме того, чтобы несколько процессов могли совместно работать с общим сегментом, они должны "договориться" о выборе одинакового ключа.
5.2. Семафоры для процессов
Как говорилось в предыдущем разделе, процессы должны координировать свои усилия при совместном доступе к памяти. Вспомните: в разделе 4.4.5, "Обычные потоковые семафоры", рассказывалось о семафорах, которые являются счетчиками, позволяющими синхронизировать работу потоков. В Linux имеется альтернативная реализация семафоров (иногда называемых семафорами System V), предназначенных для синхронизации процессов. Такие семафоры выделяются, используются и освобождаются подобно совместно используемым сегментам памяти. Для большинства случаев достаточно одного семафора, тем не менее они работают группами. В этом разделе мы опишем системные вызовы, позволяющие реализовать двоичный семафор.
5.2.1. Выделение и освобождение семафоров
Функции semget()
и semctl()
выделяют и освобождают семафоры, функционируя подобно функциям shmget()
и shmctl()
. Первым аргументом функции semget()
является ключ, идентифицирующий группу семафоров; второй аргумент — это число семафоров в группе; третий аргумент — флаги прав доступа, как в функции shmget()
. Функция semget()
возвращает идентификатор группы семафоров. Если задан ключ, принадлежащий существующей группе, будет возвращен ее идентификатор. В этом случае второй аргумент (число семафоров) может равняться нулю.
Семафоры продолжают существовать даже после того, как все работавшие с ними процессы завершились. Чтобы система не исчерпала лимит семафоров, последний процесс должен явно удалить группу семафоров. Для этого нужно вызвать функцию semctl()
, передав ей идентификатор группы, число семафоров в группе, флаг IPC_RMID
и произвольное значение типа union semun
(оно игнорируется). Значение EUID (эффективный идентификатор пользователя) процесса, вызвавшего функцию, должно совпадать с аналогичным значением процесса, создавшего группу семафоров (либо вызывающий процесс должен быть запущен пользователем root
). В отличие от совместно используемых сегментов памяти, удаляемая группа семафоров немедленно освобождается.
В листинге 5.2 представлены функции, выделяющие и освобождающие двоичный семафор.
#include
#include
#include
/* Тип union semun необходимо определить самостоятельно. */
union semun {
int val;
struct semid_ds *buf;
unsigned short int* array;
struct seminfo *__buf;
};
/* Получаем идентификатор семафора и создаем семафор,
если идентификатор оказывается уникальным. */
int binary_semaphore_allocation(key_t key, int sem_flags) {
return semget(key, 1, sem_flags);
}
/* Освобождаем семафор, подразумевая, что пользователи
больше не работают с ним. В случае ошибки
возвращается -1. */
int binary_semaphore_deallocate(int semid) {
union semun ignored_argument;
return semctl(semid, 1, IPC_RMID, ignored_argument}
}
5.2.2. Инициализация семафоров
Выделение и инициализация семафора — две разные операции. Чтобы проинициализировать семафор, вызовите функцию semctl()
, задав второй аргумент равным нулю, а третий аргумент — равным константе SETALL
. Четвертый аргумент должен иметь тип union semun
, поле array
которого указывает на массив значений типа unsigned short
. Каждое значение инициализирует один семафор из набора.
В листинге 5.3 представлена функция, инициализирующая двоичный семафор.
#include
#include
#include
/* Тип union semun необходимо определить самостоятельно. "/
union semun {
int val;
struct semid_ds* buf;
unsigned short int *array;
struct seminfo *__buf;
};
/* Инициализация двоичного семафора значением 1. */
int binary_semaphore_initialize(int semid) {
union semun argument;
unsigned short values(1);
values[0] = 1;
argument.array = values;
return semctl(semid, 0, SETALL, argument);
}
5.2.3. Операции ожидания и установки
Каждый семафор имеет неотрицательное значение и поддерживает операции ожидания и установки. Системный вызов semop()
реализует обе операции. Первым аргументом функции является идентификатор группы семафоров. Второй аргумент — это массив значений типа struct sembuf
, задающих выполняемые операции. Третий аргумент — это длина массива.
Интервал:
Закладка: