2 * Leandro Lucarella (77891)
4 * Ejercicio 1.2. Implementa problema del museo usando variables (shared memory)
12 #include <sys/types.h>
17 /// Clave de nuestro segmento shm y semaforo
18 #define SHM_KEY 0x77891120
19 #define SEM_KEY 0x77891121
21 /// Ahora el contador es un simple unsigned
22 typedef unsigned Molinete;
24 /// Hace un wait (P) al semaforo (devuelve false si hubo error).
25 void sem_wait(int sem_id)
28 op.sem_op = -1; // sacar 1
29 op.sem_num = 0; // al semaforo 1
30 op.sem_flg = 0; // esperando
31 if (semop(sem_id, &op, 1) < 0)
33 std::cerr << "No se pudo sacar al semaforo.\n";
38 /// Hace un signal (V) al semaforo (devuelve false si hubo error).
39 void sem_signal(int sem_id)
42 op.sem_op = 1; // agregar 1
43 op.sem_num = 0; // al semaforo 1
44 op.sem_flg = 0; // esperando
45 if (semop(sem_id, &op, 1) < 0)
47 std::cerr << "No se pudo poner en el semaforo.\n";
52 int main(int argc, char *argv[])
59 cerr << "Faltan parametros: " << argv[0] << " N [max_iter]\n";
62 int proc = atoi(argv[1]);
68 shm_id = shmget(SHM_KEY, sizeof(Molinete),
69 (proc ? 0 : IPC_CREAT) | 0666); // crea solo si es el proceso 0
71 while (proc != 0 && shm_id == -1);
72 Molinete* molinete = (Molinete*) shmat(shm_id, NULL, 0);
73 if (molinete == (Molinete*) -1)
75 cerr << "Error al attachear shared memory.\n";
78 cout << "Shared memory id = " << shm_id << "\n";
84 sem_id = semget(SEM_KEY, 1,
85 (proc ? 0 : IPC_CREAT) | 0666); // crea solo si es el proceso 0
87 while (proc != 0 && sem_id == -1);
89 // Si es el primero, inicializo
93 if (semctl(sem_id, 0, SETVAL, 1) < 0) // pongo 1 en el sem
95 cerr << "No se pudo inicializar semaforo.\n";
98 cout << "Semaforo inicializado id = " << sem_id << "\n";
103 // Maxima cantidad de iteraciones (puede venir por parametro)
106 max_iter = atoi(argv[2]);
109 for (int i = 0; i < max_iter; ++i)
111 char sdec[] = "sale";
112 char sinc[] = "entra";
114 bool dec = rand() % 2;
116 sem_wait(sem_id); // lock
117 if (dec && *molinete) // Decremento sólo si no es 0
127 sem_signal(sem_id); // unlock
128 // Uso cout directamente porque es line-buffered, mientras que
129 // no use threads no es problema el buffer de cout. Y de todas
130 // formas pongo un flush (el endl es \n+flush) por las dudas.
131 cout << "Proceso " << proc << " (" << s << "): molinete = " << count
133 sched_yield(); // Para ver como se entrelazan mejor
136 if (shmdt(molinete) == -1)
138 cerr << "Error al detachear shared memory.\n";
145 // vim: set et sw=4 sts=4 :