Semáforo (programación)

Un  semáforo es una primitiva de sincronización [1] del trabajo de procesos e hilos , que se basa en un contador, sobre el cual se pueden realizar dos operaciones atómicas : aumentar y disminuir el valor en uno, mientras que la operación de disminución para el valor cero de el contador está bloqueando [2 ] . Sirve para construir mecanismos de sincronización más complejos [1] y se utiliza para sincronizar tareas que se ejecutan en paralelo, para proteger la transferencia de datos a través de la memoria compartida , para proteger secciones críticas y también para controlar el acceso al hardware.

Los semáforos computacionales se utilizan para controlar recursos limitados [3] . Los semáforos binarios proporcionan la exclusión mutua de la ejecución de secciones críticas [4] , y su implementación simplificada es mutex , cuyo uso es más limitado [5] . Además de la exclusión mutua en el caso general, los semáforos y los mutex se pueden usar en muchos otros algoritmos típicos, incluida la señalización de otras tareas [6] , lo que permite que solo una tarea pase ciertos puntos de control a la vez, por analogía con un torniquete [7 ] , el problema del productor y consumidor, que implica la transferencia de datos de una tarea a otra [8] , barreras que permiten sincronizar grupos de tareas en ciertos puntos de control [9] , variables de condición para notificar a otras tareas de cualquier evento [3] y bloqueos de lectura y escritura que permiten la lectura simultánea de datos, pero prohíben su cambio simultáneo [10] .

Los problemas típicos del uso de semáforos son el bloqueo simultáneo de dos tareas mientras se esperan mutuamente [11] y el agotamiento de recursos, como resultado de lo cual un recurso puede no estar disponible periódicamente para algunas tareas debido a su uso por otras tareas [12] . Cuando se usa en procesos en tiempo real, puede ocurrir una inversión de prioridad, lo que puede causar que un proceso de mayor prioridad se bloquee indefinidamente debido a que el proceso de menor prioridad adquiere el semáforo, mientras que el tiempo de CPU se otorga al proceso de prioridad media [13] , la solución a que es la herencia de prioridad [14] .

Información general

El concepto de semáforo fue introducido en 1965 por el científico holandés Edsger Dijkstra [15] , y en 1968 propuso utilizar dos semáforos para resolver el problema del productor y el consumidor [8] .

Un semáforo es un contador en el que se pueden realizar dos operaciones: aumentar en 1 ( inglés  arriba ) y disminuir en 1 ( inglés  abajo ). Al intentar disminuir un semáforo cuyo valor es cero, la tarea que solicitó esta acción debe bloquearse hasta que sea posible reducir el valor del semáforo a un valor no negativo, es decir, hasta que otro proceso aumente el valor del semáforo [ 16] . El bloqueo de una tarea se entiende como un cambio en el estado de un proceso o subproceso por parte del programador de tareas de tal manera que la tarea suspenderá su ejecución [17] .

Las operaciones de disminuir y aumentar el valor de un semáforo se denotaban originalmente con las letras P (del holandés  proberen  - probar) y V (del holandés  verhogen  - elevar más alto), respectivamente. Dijkstra dio estas notaciones a las operaciones con semáforos, pero como las personas que hablan otros idiomas no las entienden, en la práctica se suelen utilizar otras notaciones. Las designaciones upy downse utilizaron por primera vez en el lenguaje Algol 68 [18] .

Las operaciones de incremento y decremento de un semáforo, junto con todas las comprobaciones, deben ser atómicas . Si al momento de aumentar el valor del semáforo hay más de un proceso bloqueado en este semáforo, entonces el sistema operativo selecciona uno de ellos y le permite completar la operación de disminuir el valor del semáforo [16] .

Generalmente se acepta que el valor de un semáforo no es negativo, pero existe otro enfoque para definir un semáforo, en el que un valor negativo se entiende como el número de tareas bloqueadas con signo negativo. Con este enfoque, la disminución del semáforo se bloquea si el resultado de la operación se vuelve negativo [17] .

El objetivo principal del semáforo es permitir o prohibir temporalmente la realización de cualquier acción, por lo que si el valor del contador del semáforo es mayor que cero, entonces dicen que está en un estado de señal, si el valor es cero - en un estado sin señal [19] . Reducir el valor de un semáforo también se denomina a veces adquisición ( ing. adquirir [20] ), y aumentar el valor - liberación o liberación ( ing. release [20] ) [21] , lo que hace posible hacer la descripción de el funcionamiento de un semáforo más comprensible en el contexto de controlar el uso de algún recurso o cuando se utiliza en tramos críticos.   

En general, un semáforo se puede representar como un objeto que consta de [22] :

El concepto de semáforo es muy adecuado para sincronizar hilos, puede usarse para sincronizar procesos, pero es completamente inadecuado para sincronizar la interacción de las computadoras. Un semáforo es una primitiva de sincronización de bajo nivel, por lo que, excepto para proteger secciones críticas, puede ser complicado de usar por sí solo [23] . Otra primitiva de sincronización de nivel inferior es futex . Puede ser proporcionado por el sistema operativo y es muy adecuado para implementar semáforos a nivel de aplicación cuando se utilizan operaciones atómicas en un contador compartido [24] .

Tipos de semáforos

Los semáforos pueden ser binarios y computacionales [3] . Los semáforos informáticos pueden tomar valores enteros no negativos y se utilizan para trabajar con recursos, cuyo número es limitado [3] , o participar en la sincronización de tareas de ejecución paralela. Los semáforos binarios solo pueden tomar los valores 0 y 1 [3] y se utilizan para excluir mutuamente dos o más procesos de estar en sus secciones críticas al mismo tiempo [4] .

Los semáforos mutex [3] ( mutexes ) son una implementación simplificada de semáforos, similar a los semáforos binarios con la diferencia de que los mutex deben ser liberados por el mismo proceso o subproceso que los adquiere [25] , sin embargo, dependiendo del tipo y la implementación, un intentar liberar por otro hilo puede cómo liberar el mutex y devolver un error [26] . Junto con los semáforos binarios, se utilizan para organizar secciones críticas de código [27] [28] . A diferencia de los semáforos binarios, el estado inicial de un mutex no se puede capturar [29] y pueden soportar la herencia de prioridad [30] .

Los semáforos ligeros son semáforos que utilizan un bucle de espera activo antes de ejecutar un bloqueo. Un ciclo de espera activo en algunos casos le permite reducir el número de llamadas al sistema [3] .

Algoritmos para usar

Algoritmos típicos

Señalización

La señalización, también llamada notificación, es el propósito básico de los semáforos, asegura que una pieza de código en una tarea se ejecute después de que se ejecute una pieza de código en otra tarea [6] . Señalizar el uso de un semáforo generalmente implica establecer su valor inicial en 0 para que las tareas que esperan el estado señalado puedan bloquearse hasta que ocurra el evento. La señalización se realiza incrementando el valor del semáforo, y la espera se realiza decrementando el valor [29] .

Ejemplo de señalización de semáforo
convencional
  • Inicializar semáforo A (A ← 0)
Corriente 1 Corriente 2
  • Realizar la preparación de recursos
  • Señal con semáforo A (A ← 1)
Desbloqueo de flujo 2
  • Acciones en un recurso compartido
El subproceso 2 obtuvo el tiempo de CPU primero
  • Esperar al estado de señal A (bloqueo)
Desbloquear, A ← 0
  • Acciones en un recurso compartido

Los semáforos son muy adecuados para señalar una o más tareas, cuyo número se conoce de antemano. Si no se conoce de antemano el número de tareas que esperan un estado de señal, generalmente se utilizan variables de condición .

Exclusión mutua

En las aplicaciones de subprocesos múltiples, a menudo se requiere que las secciones separadas del código, denominadas secciones críticas , no puedan ejecutarse en paralelo, por ejemplo, al acceder a algún recurso no compartido o al cambiar las ubicaciones de memoria compartida. Para proteger dichas áreas, puede usar un semáforo binario o un mutex [3] . Un mutex es más seguro de usar porque solo puede ser liberado por el proceso o subproceso que lo adquirió [5] . Además, usar un mutex en lugar de un semáforo puede ser más eficiente debido a la optimización para dos valores a nivel de implementación del código ensamblador.

El valor inicial del semáforo se establece en uno, lo que significa que no se captura; nadie ha ingresado a la sección crítica todavía. La entrada ( inglés  enter ) en la sección crítica es la captura del semáforo: su valor se reduce a 0, lo que hace un intento repetido de ingresar al bloqueo de la sección crítica. Al salir ( ing.  dejar ) de la sección crítica, el semáforo se libera y su valor pasa a ser igual a 1, lo que permite volver a ingresar a la sección crítica, incluidos otros hilos o procesos .

Para diferentes recursos, puede haber diferentes semáforos responsables de las secciones críticas. Así, las secciones críticas protegidas por diferentes semáforos pueden trabajar en paralelo.

Un ejemplo de una sección crítica basada en un semáforo
convencional
  • Inicializar semáforo A (A ← 1)
Corriente 1 Corriente 2
El subproceso 1 obtuvo el tiempo de CPU primero
  • Captura semáforo A (A ← 0)
  • Realizar acciones en un recurso
  • Soltar semáforo A (A ← 1)
Desbloqueo de flujo 2
Una capturada en la corriente 1
  • Aprovechar el semáforo A (bloqueo)
Desbloquear, A ← 0
  • Realizar acciones en un recurso
  • Soltar semáforo A (A ← 1)

Además de los semáforos, la exclusión mutua se puede organizar a través de otros métodos de sincronización, por ejemplo, a través de monitores , si son compatibles con el lenguaje de programación utilizado. Los monitores le permiten proteger un conjunto de datos ocultando detalles de sincronización del programador y brindando acceso a datos protegidos solo para monitorear procedimientos, y la implementación de monitores se deja al compilador y generalmente se basa en un mutex o un semáforo binario. En comparación con los semáforos, los monitores pueden reducir la cantidad de errores en los programas, pero a pesar de la facilidad de uso, la cantidad de monitores compatibles con idiomas es pequeña [31] .

Torniquete

A menudo es la tarea de permitir o denegar el paso de una o más tareas a través de ciertos puntos de control. Para solucionar este problema se utiliza un algoritmo basado en dos semáforos, que en su funcionamiento se asemeja a un torniquete, ya que permite saltarse una sola tarea a la vez. El torniquete se basa en un semáforo, que se captura en los puntos de control y se libera de inmediato. Si se requiere cerrar el torniquete, entonces se debe tomar el semáforo, como resultado de lo cual se bloquearán todas las tareas que pasen por el torniquete. Si desea permitir que las tareas vuelvan a pasar por el torniquete, basta con soltar el semáforo, después de lo cual las tareas continuarán ejecutándose por turnos [7] .

Pasar alternativamente por el torniquete tiene un gran inconveniente: para cada paso, puede ocurrir un cambio de contexto innecesario entre tareas, como resultado de lo cual el rendimiento del algoritmo disminuirá. En algunos casos, la solución puede ser utilizar un torniquete de varios asientos que desbloquea varias tareas a la vez, lo que se puede hacer, por ejemplo, liberando cíclicamente el semáforo si la implementación del semáforo utilizada no admite un aumento en un número arbitrario [ 32] .

Pseudocódigo de torniquete
Inicialización Torniquete bloqueando desbloquear
torniquete = Semáforo(1) apoderarse (torniquete) soltar (torniquete) apoderarse (torniquete) soltar (torniquete)

Los torniquetes basados ​​en semáforos se pueden utilizar, por ejemplo, en mecanismos de barrera [33] o bloqueos de lectura/escritura [34] .

Cambiar

Otro algoritmo típico basado en semáforos es la implementación del interruptor. Las tareas pueden agarrar el interruptor y soltarlo. La primera tarea que agarra el interruptor es encenderlo. Y la última tarea que lo libera lo apaga. Para este algoritmo, podemos dibujar una analogía con un interruptor de luz en una habitación. El primero en entrar a la habitación enciende la luz, y el último en salir la apaga [35] .

El algoritmo se puede implementar en base al contador de tareas que capturaron el interruptor y el semáforo del interruptor, cuyas operaciones deben estar protegidas por un mutex. Cuando se captura el interruptor, el contador se incrementa en 1, y si su valor ha cambiado de cero a uno, entonces se captura el semáforo del interruptor, lo que equivale a encender el interruptor. En este caso, incrementar el contador, junto con verificar y capturar el semáforo, son una operación atómica protegida por un mutex. Cuando se suelta el interruptor, el contador disminuye, y si su valor llega a cero, entonces se suelta el semáforo del interruptor, es decir, el interruptor cambia al estado de apagado. Disminuir el contador junto con verificarlo y liberar el semáforo también debe ser una operación atómica [35] .

Pseudocódigo del algoritmo de operación del interruptor automático
Tipo de datos Inicialización Uso
Cambiar: cuenta = 0 mutex = semáforo (1) Cambiar, lock(objetivo-semáforo): agarrar (mutex) cantidad += 1 si cuenta == 1: capturar (objetivo-semáforo) liberar (mutex) Cambiar, desbloquear (objetivo-semáforo): agarrar (mutex) cantidad -= 1 si cuenta == 0: liberación (objetivo-semáforo) liberar (mutex) interruptor = interruptor () semáforo = semáforo(1) bloque (interruptor, semáforo) // Sección crítica del interruptor, // el semáforo está bloqueado desbloquear (interruptor, semáforo)

El algoritmo de cambio se utiliza en un mecanismo más complejo: bloqueos de lectura y escritura [35] .

El problema del productor y el consumidor

La tarea de productor consumidor implica la producción de alguna información por parte de una tarea y la transferencia de esta información a otra tarea para su procesamiento. En los sistemas de subprocesos múltiples, la producción y el consumo simultáneos pueden generar condiciones de carrera , lo que requiere el uso de secciones críticas u otros medios de sincronización. El semáforo es la primitiva de sincronización más simple que se puede utilizar para resolver el problema del productor y el consumidor.

Pasar datos a través de un búfer de anillo

El búfer de anillo es un búfer con un número fijo de elementos, en el que los datos se ingresan y procesan en una base FIFO (primero en entrar, primero en salir ). En una versión de un solo subproceso, 4 celdas de memoria son suficientes para organizar dicho búfer:

  • el número total de elementos en el búfer,
  • el número de elementos ocupados o libres en el búfer,
  • número ordinal del elemento actual,
  • el número ordinal del siguiente elemento.

En una implementación multitarea, el algoritmo se complica por la necesidad de sincronizar tareas. Para el caso de dos tareas (productor y consumidor), podemos limitarnos a dos celdas de memoria y dos semáforos [8] :

  • el índice del siguiente elemento legible,
  • el índice del siguiente elemento de escritura,
  • un semáforo que permite leer el siguiente elemento,
  • un semáforo que permite escribir el siguiente elemento libre del búfer.

El valor inicial del semáforo responsable de la lectura se establece en 0 porque la cola está vacía. Y el valor del semáforo responsable de la escritura se establece igual al tamaño total del búfer, es decir, todo el búfer está disponible para llenarse. Antes de llenar el siguiente elemento en el búfer, el semáforo de escritura se decrementa en 1, reservando el siguiente elemento de la cola para escribir datos, después de lo cual se cambia el índice de escritura y el semáforo de lectura aumenta en 1, lo que permite leer el elemento agregado. a la cola La tarea de lectura, por el contrario, captura el semáforo para lectura, después de lo cual lee el siguiente elemento del búfer y cambia el índice del siguiente elemento para lectura, y luego libera el semáforo para escritura, lo que permite que la tarea de escritura escriba en el elemento liberado [8] .

Pseudocódigo de búfer de anillo
Inicialización Uso
tamaño de búfer = N permiso de escritura = Semáforo (tamaño de búfer) permiso de lectura = Semáforo (0) por escritura = 0 en lectura = 0 búfer = matriz (tamaño del búfer) // Tarea de escritura elemento-producido = elemento-producido() capturar (permiso de escritura) buffer[por-escritura] = elemento producido por escritura += 1 si por registro >= tamaño de búfer: por escritura = 0 liberación (permiso de lectura) // Leer tarea agarrar (permiso de lectura) lectura-elemento = búfer[por-lectura] por lectura += 1 si por lectura >= tamaño del búfer: en lectura = 0 liberación (permiso de escritura) proceso (elemento de lectura)

Si se implementa un búfer de anillo para varios escritores y lectores, se agrega un mutex a la implementación que bloquea el búfer cuando se escribe o se lee [36] .

Pasar datos a través de un búfer arbitrario

Además de transferir datos a través de un búfer de anillo, también es posible transferir a través de uno arbitrario, pero en este caso, los datos de escritura y lectura deben estar protegidos por un mutex, y el semáforo se usa para notificar la tarea de lectura sobre la presencia del siguiente elemento en el búfer. La tarea de escritura agrega un elemento protegido por el mutex al búfer y luego señala su presencia. La tarea de lectura captura el semáforo y luego, bajo la protección del mutex, recibe el siguiente elemento. Vale la pena mencionar que intentar adquirir un semáforo protegido por exclusión mutua puede provocar un punto muerto si se intenta leer desde un búfer vacío, y liberar el semáforo dentro de una sección crítica puede degradar ligeramente el rendimiento. Este algoritmo, como en el caso de un búfer de anillo protegido por un mutex, permite que varias tareas escriban y lean simultáneamente [37] .

En los mecanismos de sincronización

Barrera

Una barrera es un mecanismo para sincronizar puntos críticos para un grupo de tareas. Las tareas solo pueden atravesar la barrera todas a la vez. Antes de entrar en un punto crítico, las tareas de un grupo deben bloquearse hasta que la última tarea del grupo alcance el punto crítico. Una vez que todas las tareas están a punto de entrar en sus puntos críticos, deben continuar con su ejecución [9] .

La solución más sencilla para organizar una barrera en el caso de dos tareas se basa en dos semáforos binarios A y B, inicializados a cero. En el punto crítico de la primera tarea, se debe señalizar el semáforo B y luego se debe capturar el semáforo A. En el punto crítico de la segunda tarea, primero se debe señalizar el semáforo A y luego se debe capturar B. señalará otra tarea , permitiendo su ejecución. Una vez que ambas tareas hayan llegado a sus puntos críticos, se señalarán sus semáforos, lo que les permitirá continuar con su ejecución [38] .

Pseudocódigo de barrera simple
Inicialización Tarea usando la barrera
cantidad objetivo = N cuenta = 0 mutex = semáforo (1) entrada-torniquete = Semáforo(0) // Primera fase de barrera agarrar (mutex) cantidad += 1 if cuenta == cuenta-tareas: liberación (entrada-torniquete) liberar (mutex) apoderarse (entrada-torniquete) liberación (entrada-torniquete) // Punto crítico

Tal implementación es de un solo paso, ya que la barrera no vuelve a su estado original, también tiene bajo rendimiento debido al uso de un torniquete de un solo asiento, lo que requiere un cambio de contexto para cada tarea, por lo que esta solución es de poca utilidad. uso en la práctica [32] .

Barrera Bifásica

Una característica de la barrera de dos fases es que, al usarla, cada tarea se detiene en la barrera dos veces, antes del punto crítico y después. Dos topes hacen que la barrera sea reentrante , ya que el segundo tope permite que la barrera vuelva a su estado original [39] .

El algoritmo de reentrada universal del mecanismo de barrera de dos fases se puede basar en el uso de un contador de tareas que han llegado al punto crítico y dos torniquetes de varios asientos. Las operaciones en el mostrador y el control de los torniquetes deben estar protegidos por un mutex. En este caso, el número total de tareas debe conocerse de antemano. El primer torniquete permite pasar las tareas al punto crítico y debe bloquearse inicialmente. El segundo omite tareas que acaban de pasar el punto crítico y también debe bloquearse inicialmente. Antes de acercarse al punto crítico, el contador de tareas alcanzadas se incrementa en 1, y tan pronto como llega al número total de tareas, se desbloquea el primer torniquete para todas las tareas, pasándolas al punto crítico, lo que sucede atómicamente a través del mutex. junto con el incremento del contador y su verificación. Después del punto crítico, pero antes del segundo torniquete, el contador del número de tareas se reduce en 1. Cuando el valor llega a cero, el segundo torniquete se desbloquea para todas las tareas, mientras que las operaciones en el segundo torniquete también ocurren atómicamente, junto con el contador decremento y su control. Como resultado, todas las tareas se detienen primero antes del punto crítico y luego después. Después de pasar la barrera, los estados del contador y torniquetes están en sus valores originales [32] .

Pseudocódigo del Algoritmo de Barrera de Dos Fases Reentrantes
Inicialización Tarea usando la barrera
mutex = semáforo (1) cuenta = 0 entrada-torniquete = Semáforo(0) torniquete de salida = Semáforo (0) // Primera fase de barrera agarrar (mutex) cantidad += 1 if cuenta == cuenta-tareas: release(entrada-torniquete, cantidad) liberar (mutex) apoderarse (entrada-torniquete) // Punto crítico // Segunda fase de barrera agarrar (mutex) cantidad -= 1 si cuenta == 0: release(salida-torniquete, cantidad) liberar (mutex) apoderarse (torniquete de salida)
Variable de condición

Una variable de condición es una forma de notificar tareas pendientes cuando ocurre un evento [3] . El mecanismo de variable de condición a nivel de aplicación generalmente se basa en un futex y proporciona funciones para esperar un evento y enviar una señal sobre su ocurrencia, pero partes separadas de estas funciones deben estar protegidas por un mutex o semáforo, ya que además del futex, el mecanismo de variable de condición generalmente contiene datos compartidos adicionales [40] . En implementaciones simples, el futex puede ser reemplazado por un semáforo, el cual, cuando sea notificado, deberá ser liberado tantas veces como el número de tareas suscritas a la variable de condición, sin embargo, con una gran cantidad de suscriptores, la notificación puede volverse un cuello de botella [41] .

El mecanismo de variable de condición asume la presencia de tres operaciones: esperar un evento, señalar un evento a una tarea y notificar a todas las tareas sobre un evento. Para implementar un algoritmo basado en semáforos, necesitará: un mutex o un semáforo binario para proteger la variable de condición en sí, un contador para el número de tareas en espera, un mutex para proteger el contador, un semáforo A para bloquear las tareas en espera y un semáforo B adicional para activar la siguiente tarea en espera a tiempo [42] .

Al suscribirse a eventos, el contador de tareas suscritas se incrementa atómicamente en 1, después de lo cual se libera el mutex precapturado de la variable de condición. Luego se captura el semáforo A para esperar a que ocurra el evento. Al ocurrir un evento, la tarea de señalización verifica atómicamente el contador de tareas suscritas y notifica a la siguiente tarea de la ocurrencia del evento liberando el semáforo A, y luego bloquea en el semáforo B, esperando la confirmación de desbloqueo. La tarea alertada libera el semáforo B y vuelve a adquirir el mutex de la variable de condición para volver a su estado original. Si se realiza una notificación de difusión de todas las tareas suscritas, entonces el semáforo A de tareas bloqueadas se libera en un ciclo de acuerdo con la cantidad de tareas suscritas en el contador. En este caso, la notificación ocurre atómicamente bajo la protección del contador mutex, de modo que el contador no puede cambiar durante la notificación [42] .

Pseudocódigo de variable de condición
Declaración de tipos Uso
variable-condición(): cuenta = 0 mutex = semáforo (1) evento de espera = Semáforo (0) evento-recepción = Semáforo(0) variable condicional, esperar (objetivo-mutex): agarrar (mutex) cantidad += 1 liberar (mutex) liberación (objetivo-mutex) agarrar (esperar eventos) lanzamiento (obtener eventos) agarrar (objetivo-mutex) variable condicional, notificar(): agarrar (mutex) si cantidad > 0: cantidad -= 1 liberación (eventos de espera) agarrar (obtener eventos) liberar (mutex) variable condicional, visitar-todos(): agarrar (mutex) si cantidad > 0: liberación (esperar eventos, contar) agarrar (obtener eventos, contar) cuenta = 0 liberar (mutex) // inicialización evento = variable-condición() mutex = semáforo (1) // Esperar un evento agarrar (mutex) esperar (evento) // Sección crítica del evento liberar (mutex) // Alertar una tarea notificar (evento) // Notificar todas las tareas notificar a todos (evento)

La solución del semáforo tiene un problema importante: dos cambios de contexto de señalización, lo que reduce en gran medida el rendimiento del algoritmo, por lo que, al menos a nivel de los sistemas operativos, generalmente no se usa [42] .

Un hecho interesante es que el propio semáforo se implementa fácilmente en función de una variable de condición y un mutex [24] , mientras que la implementación de una variable de condición basada en semáforos es mucho más complicada [42] .

Bloqueos de lectura y escritura

Uno de los problemas clásicos es la sincronización del acceso a un recurso que está disponible para leer y escribir al mismo tiempo. Los bloqueos de lectura y escritura están diseñados para resolver este problema y le permiten organizar bloqueos de lectura y escritura separados en un recurso, lo que permite la lectura simultánea, pero prohíbe la escritura simultánea. La escritura también bloquea cualquier lectura [10] . No se puede construir un mecanismo eficiente sobre la base de un futex solo, el contador del número de lectores puede cambiar sin desbloquear ninguna tarea [24] . Los bloqueos de lectura y escritura se pueden implementar en función de una combinación de exclusión mutua y semáforos, o exclusión mutua y una variable de condición.

El algoritmo universal, desprovisto del problema de escasez de recursos de las tareas de escritura, incluye un interruptor de semáforo binario A para organizar una sección crítica de tareas de lectura y un torniquete para bloquear nuevas tareas de lectura en presencia de escritores en espera. Cuando llega la primera tarea para leer, toma el semáforo A con un interruptor, evitando escrituras. Para los escritores, el semáforo A protege la sección crítica del escritor, por lo que si los lectores lo capturan, todos los escritores se bloquean al ingresar a su sección crítica. Sin embargo, la captura por tareas de escritor del semáforo A y posterior escritura está protegida por el semáforo torniquete. Por lo tanto, si se produce un bloqueo de una tarea de escritura por la presencia de lectores, el torniquete se bloquea junto con nuevas tareas de lectura. Tan pronto como el último lector termina su trabajo, se libera el semáforo del interruptor y se desbloquea el primer escritor en la cola. Al final de su trabajo, libera el semáforo del torniquete, permitiendo nuevamente el trabajo de tareas de lectura [34] .

Pseudocódigo del algoritmo universal de bloqueo de lectura y escritura
Inicialización tarea de lectura Tarea de escritura
interruptor = interruptor () permiso de escritura = Semáforo (1) torniquete = Semáforo(1) apoderarse (torniquete) liberación (torniquete) lock(interruptor, permiso-escritura) // Sección crítica de la tarea de lectura desbloquear (cambiar, permiso de escritura) apoderarse (torniquete) capturar (permiso de escritura) // Sección crítica de la tarea de escritura soltar (torniquete) liberación (permiso de escritura)

A nivel de sistemas operativos existen implementaciones de semáforos de lectura y escritura, los cuales son modificados de manera especial para aumentar la eficiencia en uso masivo [43] .

En problemas clásicos

Filósofos gastronómicos

Uno de los problemas clásicos de sincronización es el Problema de los Filósofos Comedores. El problema incluye 5 filósofos cenando en una mesa redonda, 5 platos, 5 tenedores y un plato de pasta compartido en el medio de la mesa. Hay un plato frente a cada filósofo y un tenedor a derecha e izquierda, pero cada tenedor se comparte entre dos filósofos vecinos y solo se puede comer pasta con dos tenedores a la vez. Además, cada uno de los filósofos puede pensar o comer pasta [44] .

Los filósofos representan los hilos que interactúan en el programa, y ​​la solución del problema incluye una serie de condiciones [44] :

  • no debe haber puntos muertos entre los filósofos ;
  • ningún filósofo debe morir de hambre mientras espera la liberación del tenedor ;
  • debería ser posible que al menos dos filósofos comieran al mismo tiempo.

Para resolver el problema, a cada bifurcación se le puede asignar un semáforo binario. Cuando el filósofo intenta coger el tenedor, se captura el semáforo, y en cuanto termina de comer se sueltan los semáforos de los tenedores. El problema es que el vecino ya pudo tomar el tenedor, entonces el filósofo queda bloqueado hasta que su vecino come. Si todos los filósofos empiezan a comer al mismo tiempo, es posible un punto muerto [44] .

Una solución al punto muerto podría ser limitar el número de filósofos que comen al mismo tiempo a 4. En este caso, al menos un filósofo podrá cenar mientras los demás esperan. La restricción se puede implementar a través de un semáforo con un valor inicial de 4. Cada uno de los filósofos capturará este semáforo antes de tomar los tenedores, y después de comer, lo soltará. Además, esta solución garantiza que los filósofos no pasarán hambre, porque si un filósofo espera a que un vecino suelte el tenedor, tarde o temprano lo soltará [44] .

También hay una solución más sencilla. El punto muerto es posible si 5 filósofos sostienen simultáneamente un tenedor en la misma mano, por ejemplo, si todos son diestros y tomaron primero el tenedor derecho. Si uno de los filósofos es zurdo y toma primero la bifurcación izquierda, entonces no es posible ni el punto muerto ni el hambre. Así, basta con que uno de los filósofos capte primero el semáforo de la bifurcación izquierda, y luego el de la derecha, mientras que los demás filósofos hacen lo contrario [44] .

Montaña rusa

Otro problema clásico es el problema de la montaña rusa , en el que un tren de carritos se llena completamente de pasajeros, luego los hace rodar y regresa por más. De acuerdo con las condiciones del problema, el número de pasajeros dispuestos excede el número de asientos en el tren, por lo que los próximos pasajeros esperan en fila mientras el tren da vueltas. Si el tren tiene M asientos, entonces el tren primero debe esperar hasta que M pasajeros se sienten en sus asientos, luego debe llevarlos, esperar hasta que todos bajen y nuevamente esperar nuevos pasajeros [45] .

La composición de los carros junto con los pasajeros se puede representar como tareas interactivas. Cada pasajero debe bloquearse mientras espera su turno, y el propio tren debe bloquearse en las etapas de llenado y vaciado de asientos. Para cargar y descargar el tren, puede usar dos semáforos con interruptores, cada uno protegido por su propio mutex, y para bloquear pasajeros para cargar y descargar, puede usar dos semáforos responsables de lugares en los carros. Los pasajeros que esperan toman el semáforo de carga, y el tren con el semáforo de carga notifica a M sobre la disponibilidad de asientos. Luego, el tren es bloqueado por un interruptor hasta que el último pasajero que aborda señale con el semáforo apropiado, después de lo cual comienza el viaje. Antes del viaje, los pasajeros son bloqueados por un semáforo para la descarga, lo que les impide salir del tren. Después del viaje, el tren notifica a M pasajeros con un semáforo de descarga, permitiéndoles bajar, y luego bloquea en el semáforo del interruptor para descargar, esperando hasta que todos los pasajeros se hayan ido. Tan pronto como el último pasajero abandone el tren, señalará el semáforo del segundo interruptor y permitirá que el tren vuelva a recoger pasajeros [45] .

Problemas de uso

Restricciones de semáforos

El concepto de un semáforo proporciona solo las operaciones de decremento e incremento en 1. Al mismo tiempo, una tarea que decrementa un semáforo generalmente no puede saber si se bloqueará o no. Al señalar, no hay forma de saber si hay tareas bloqueadas por el semáforo, y si una tarea señala a otro semáforo bloqueado, entonces ambas tareas continúan trabajando en paralelo y no hay forma de saber cuál de ellos recibirá tiempo de procesador siguiente [17] .

A pesar de las limitaciones del concepto de semáforos, las implementaciones específicas de los mismos pueden carecer de ciertas restricciones. Por ejemplo, la capacidad de incrementar un valor de semáforo en un número arbitrario se proporciona en las implementaciones de Linux [46] , Windows [41] y System V (POSIX) [47] . Y los semáforos POSIX le permiten determinar si se producirá un bloqueo de semáforo [48] .

Semáforos fuertes y débiles

Además de las limitaciones del propio concepto de semáforo, también existen limitaciones impuestas por el sistema operativo o una implementación particular de un semáforo. El programador de tareas del sistema operativo suele ser responsable de asignar el tiempo de procesador entre procesos y subprocesos . El uso de semáforos impone una serie de requisitos en el programador y en la propia implementación del semáforo para evitar el agotamiento de los recursos, lo que es inaceptable en aplicaciones multitarea [49] .

  1. Si hay al menos una tarea lista para ser ejecutada, debe ser ejecutada [49] .
  2. Si la tarea está lista para su ejecución, el tiempo antes de su ejecución debe ser finito [49] .
  3. Si hay una señalización de semáforo que tiene tareas bloqueadas, al menos una de ellas debe pasar al estado listo [49] .
  4. Si una tarea está bloqueada en un semáforo, entonces el número de otras tareas que se desbloquearán en el mismo semáforo antes del dado debe ser finito [49] .

Los primeros dos requisitos son necesarios para que cualquier tarea pueda obtener tiempo de procesador y no estar en un estado infinito de preparación, lo que ya le permite escribir aplicaciones sin escasez de recursos. El tercer requisito es necesario para evitar la escasez de recursos en la exclusión mutua basada en semáforos. Si la señalización solo aumentará el contador de semáforos, pero no activará la tarea bloqueada en él, entonces es posible una situación en la que la misma tarea libera y captura infinitamente el semáforo, y otras tareas bloqueadas no tienen tiempo para entrar en el estado listo, o lo hacen, pero con mucha menos frecuencia. Sin embargo, incluso si se cumple el tercer requisito, en el caso de una gran cantidad de tareas bloqueadas, es posible que se agoten los recursos si se desbloquean las mismas tareas cada vez. Este problema se resuelve con el cuarto requisito, que se observa, por ejemplo, si las tareas bloqueadas por el semáforo se despiertan a su vez [49] .

El cumplimiento de los tres primeros requisitos permite la implementación de los llamados semáforos débiles , y el cumplimiento de los cuatro - fuertes [49] .

Interbloqueos

Si los semáforos se usan incorrectamente, pueden ocurrir interbloqueos [50]  : situaciones en las que dos o más tareas paralelas se bloquean, esperando un evento entre sí [11] . En tal situación, las tareas no podrán continuar con su ejecución normalmente y, por lo general, es necesario forzar la terminación de uno o más procesos. Los interbloqueos pueden ser el resultado de un semáforo simple u otros errores de sincronización, o condiciones de carrera , que son más difíciles de depurar.

Un error común es llamar dentro de una sección crítica a una subrutina que usa la misma sección crítica [51] .

Un ejemplo ilustrativo de bloqueo mutuo pueden ser capturas anidadas de semáforos binarios A y B que protegen diferentes recursos, siempre que sean capturados en orden inverso en uno de los hilos, lo que puede deberse, por ejemplo, a diferencias de estilo en la escritura del programa. código. El error de una implementación de este tipo es una condición de carrera, que puede hacer que el programa se ejecute la mayor parte del tiempo, pero en el caso de captura de recursos en paralelo, las posibilidades de un punto muerto son altas [52] .

Ejemplo de mutex con anidamiento inverso de secciones críticas [53]
convencional
  • Inicializar semáforo A (A ← 1)
  • Inicializar semáforo B (B ← 1)
Corriente 1 Corriente 2
  • Captura semáforo A (A ← 0)
B capturado en la corriente 2
  • Aprovechar el semáforo B (bloqueo)
  • Realizar acciones en un recurso
  • Suelte el semáforo B
  • Suelte el semáforo A
  • Captura semáforo B (B ← 0)
Una capturada en la corriente 1
  • Aprovechar el semáforo A (bloqueo)
  • Realizar acciones en un recurso
  • Suelte el semáforo A
  • Suelte el semáforo B

Hambre de recursos

Similar al interbloqueo es el problema del agotamiento de los recursos, que puede no conducir a un cese total del trabajo, pero puede resultar extremadamente negativo al implementar el algoritmo. La esencia del problema radica en las negativas periódicas o frecuentes a obtener un recurso debido a su captura por otras tareas [12] .

Un caso típico para este problema es una implementación simple de bloqueos de lectura/escritura , que bloquea el recurso para escribir mientras lee. La aparición periódica de nuevas tareas de lectura puede provocar un bloqueo de escritura ilimitado en el recurso. Con poca carga en el sistema, es posible que el problema no se manifieste durante mucho tiempo; sin embargo, con mucha carga, puede surgir una situación en la que haya al menos una tarea de lectura en un momento dado, lo que hará que el bloqueo de escritura sea permanente durante el tiempo de alta carga [12] . Dado un semáforo que se libera cuando la cola de lectores está vacía, una solución simple sería agregar un semáforo binario (o mutex) para proteger el código de los escritores, mientras que al mismo tiempo actúa como un torniquete para el lectores Los escritores ingresarán a la sección crítica y tomarán un semáforo de cola vacío, bloqueando dos semáforos siempre que haya lectores. Las tareas del lector se bloquearán al ingresar al torniquete si la tarea del escritor está esperando que los lectores completen su trabajo. Tan pronto como la última tarea de lectura ha terminado su trabajo, libera el semáforo de la cola vacía, desbloqueando la tarea de escritura en espera [34] .

La exclusión mutua también puede sufrir escasez de recursos si su implementación se basa en semáforos débiles, pero existen algoritmos para eludir las limitaciones de los semáforos débiles en este caso [49] .

Inversión de Prioridad

Otro problema puede ser la inversión de prioridad que puede ocurrir cuando los procesos en tiempo real utilizan semáforos. Los procesos en tiempo real pueden ser interrumpidos por el sistema operativo solo para la ejecución de procesos con mayor prioridad. En este caso, el proceso puede bloquearse en el semáforo, esperando que sea liberado por un proceso con menor prioridad. Si en este momento se está ejecutando un proceso con una prioridad promedio entre dos procesos, entonces un proceso con una prioridad alta puede bloquearse por un período de tiempo ilimitado [13] .

El problema de la inversión de prioridades se resuelve mediante la herencia de prioridades [14] . Si es posible, los semáforos se pueden reemplazar por mutex, ya que los mutex pueden tener una herencia de precedencia predeterminada. Por lo tanto, cuando un hilo con una prioridad más alta captura un mutex, la prioridad de la tarea propietaria del mutex se incrementará de manera preventiva para liberarlo lo antes posible [30] .

La herencia ubicua de prioridades es una tarea extremadamente difícil de implementar, por lo que los sistemas que la soportan solo pueden tener una implementación parcial. Además, la herencia de prioridad crea otros problemas, como la incapacidad de combinar código con herencia de prioridad con código sin herencia cuando se utiliza la misma sección crítica [54] .

Si es necesario utilizar semáforos o si no hay soporte para la herencia de prioridades, los algoritmos pueden modificarse para aumentar de forma independiente las prioridades por tareas [54] .

Programación de aplicaciones

Semáforos en POSIX

Los estándares POSIX a nivel de sistema operativo proporcionan una API de lenguaje C para manejar semáforos tanto a nivel de hilo como a nivel de proceso a través de la memoria compartida . Los estándares definen un tipo de datos de semáforo sem_ty un conjunto de funciones para trabajar con él [55] . Los semáforos POSIX están disponibles en Linux , macOS , FreeBSD y otros sistemas operativos compatibles con POSIX.

Funciones para trabajar con semáforos POSIX desde el archivo de cabecera semaphore.h[55]
Función Descripción
sem_init()[doc. una] Inicializar un semáforo con un valor inicial para el contador y un indicador de uso a nivel de proceso.
sem_destroy()[doc. 2] Suelta el semáforo.
sem_open()[doc. 3] Cree un semáforo con nombre nuevo o abra uno existente.
sem_close()[doc. cuatro] Cerrar el semáforo después de terminar de trabajar con él.
sem_unlink()[doc. 5] Eliminar el nombre de un semáforo con nombre (no lo destruye).
sem_wait()[doc. 6] Decrementa el valor del semáforo en 1.
sem_timedwait()[doc. 7] Disminuir el valor de un semáforo en 1, con un límite en el tiempo máximo de bloque después del cual se devuelve un error.
sem_trywait()[doc. ocho] Intentar disminuir un semáforo en modo sin bloqueo devuelve un error si no es posible disminuir sin bloquear.
sem_post()[doc. 9] Aumente el valor del semáforo en 1.
sem_getvalue()[doc. diez] Obtiene el valor actual del semáforo.

Una de las desventajas de los semáforos POSIX es la especificación de la función propensa a errores sem_timedwait()que opera en el reloj de tiempo real ( CLOCK_REALTIME) [56] en lugar del tiempo de actividad del sistema ( CLOCK_MONOTONIC), lo que puede provocar que los programas se bloqueen cuando cambia la hora del sistema y puede ser crítico para los sistemas integrados. [57] , pero algunos sistemas operativos en tiempo real ofrecen análogos de esta función que funcionan con el tiempo de actividad del sistema [ 58] . Otro inconveniente es la falta de soporte para esperar en varios semáforos al mismo tiempo, o en un semáforo y un descriptor de archivo.

En Linux, los semáforos POSIX se implementan en la biblioteca Glibc basada en futex [59] .

Semáforos del sistema V

Los estándares POSIX también definen un conjunto de funciones del estándar X/Open System Interfaces (XSI) para el manejo de semáforos entre procesos dentro del sistema operativo [60] . A diferencia de los semáforos ordinarios, los semáforos XSI se pueden aumentar y disminuir en un número arbitrario, se asignan en matrices y su vida útil no se extiende a los procesos, sino al sistema operativo. Por lo tanto, si olvida cerrar el semáforo XSI cuando finalizan todos los procesos de la aplicación, seguirá existiendo en el sistema operativo, lo que se denomina fuga de recursos. En comparación con los semáforos XSI, los semáforos POSIX regulares son mucho más fáciles de usar y pueden ser más rápidos [61] .

Los conjuntos de semáforos XSI dentro del sistema se identifican mediante una clave numérica de tipo key_t, sin embargo, es posible crear conjuntos de semáforos anónimos para usar dentro de una aplicación especificando una constante IPC_PRIVATEen lugar de una clave numérica [62] .

Funciones para trabajar con semáforos XSI desde un archivo de cabecerasys/sem.h
Función Descripción
semget()[doc. once] Crea u obtiene un identificador de conjunto de semáforos con la clave numérica dada [62] .
semop()[doc. 12] Realiza operaciones atómicas de decremento e incremento por el número dado del contador de semáforos por su número del conjunto con el identificador dado, y también permite bloquear esperando el valor cero del contador de semáforos si se especifica 0 [47] como el número dado .
semctl()[doc. 13] Le permite administrar un semáforo por su número de un conjunto con un identificador dado, incluida la obtención y configuración del valor actual del contador; también responsable de destruir el conjunto de semáforos [63] .

Semáforos en Linux

Los sistemas operativos Linux admiten semáforos POSIX, pero también ofrecen una alternativa a los semáforos en forma de un contador vinculado a un descriptor de archivo a través de una llamada al sistema eventfd()[doc. 14] con bandera EFD_SEMAPHORE. Cuando dicho contador se lee a través de una función read(), se decrementa en 1 si su valor era distinto de cero. Si el valor era nulo, se produce el bloqueo (si no se especifica el indicador EFD_NONBLOCK), como ocurre con los semáforos ordinarios. La función write()incrementa el valor del contador por el número que se escribe en el descriptor de archivo. La ventaja de un semáforo de este tipo es la capacidad de esperar el estado señalado del semáforo junto con otros eventos mediante llamadas al sistema select()o poll()[46] .

Semáforos en Windows

El kernel de Windows también proporciona una API C para trabajar con semáforos. Los subprocesos bloqueados en un semáforo se ponen en cola FIFO , pero pueden llegar al final de la cola si el subproceso se interrumpe para procesar otros eventos [19] .

Funciones básicas para trabajar con semáforos API de Windows
Función Descripción
CreateSemaphoreA()[doc. quince] Cree un semáforo, especificando el valor inicial del contador, el valor máximo y el nombre del semáforo.
OpenSemaphoreW()[doc. dieciséis] Acceder a un semáforo por su nombre si ya existe.
CloseHandle()[doc. 17] Cerrar el semáforo después de terminar de trabajar con él.
WaitForSingleObject()[doc. 18] oWaitForMultipleObjects() [doc. 19] Decrementa el valor del semáforo en 1 con bloqueo en caso de valor cero del contador; le permite limitar el tiempo máximo de bloqueo.
ReleaseSemaphore()[doc. veinte] Incrementa el valor del semáforo en la cantidad especificada.

Las funciones de semáforo de Windows incluyen la capacidad de incrementar un semáforo en un número arbitrario [41] y la capacidad de esperar su estado de señal junto con el bloqueo de esperas para otros semáforos u objetos [64] .

Soporte en lenguajes de programación

Los semáforos generalmente no se admiten explícitamente en el nivel del lenguaje de programación, pero a menudo los proporcionan bibliotecas integradas o de terceros. En algunos lenguajes, como Ada [65] y Go [66] , los semáforos se implementan fácilmente en el lenguaje.

Semáforos en lenguajes de programación comunes
Idioma Módulo o biblioteca Tipo de datos
xi pthread,rt sem_t[doc. 21]
ada GNAT.Semaphores[doc. 22] Counting_Semaphore,Binary_Semaphore
C++ Boost boost::interprocess::interprocess_semaphore[doc. 23]
C# System.Threading[doc. 24] Semaphore[doc. 25]
D core.sync.semaphore[doc. 26] Semaphore[doc. 27]
Vamos golang.org/x/sync/semaphore[doc. 28] Weighted
Java java.util.concurrent[doc. 29] java.util.concurrent.Semaphore[doc. treinta]
Pitón threading[doc. 31] ,asyncio [doc. 32] threading.Semaphore[doc. 33] ,asyncio.Semaphore [doc. 34]

Ejemplos de uso

Protección de la sección crítica

El ejemplo más simple del uso de un semáforo es la exclusión mutua de la posibilidad de ejecutar secciones críticas de código para hilos o procesos. Para organizar la exclusión mutua, puede servir un semáforo binario y dos funciones: entrar en la sección crítica y salir de ella. Para simplificar, el ejemplo no incluye la capacidad de recordar la ID del subproceso de captura y la ID del proceso al que pertenece el subproceso. También se supone que la sección crítica tiene un tiempo de ejecución finito, no muy largo, por lo que EINTRse ignoran las interrupciones de la operación de captura de semáforo ( ), y los resultados de la interrupción pueden procesarse después de la sección crítica. El semáforo en sí se abstrae en una estructura para mejorar la legibilidad del código.

En el ejemplo, se lanzan dos subprocesos, uno de los cuales incrementa el contador y el otro lo decrementa. Dado que el contador es un recurso compartido, el acceso a él debe ser mutuamente excluyente; de ​​lo contrario, un subproceso puede sobrescribir los resultados de las operaciones de otro y el valor del resultado final puede ser erróneo. Por lo tanto, el contador está protegido por un semáforo binario abstracto que implementa la exclusión mutua.

Ejemplo de una implementación de sección crítica simple basada en semáforos en C (POSIX) #include <errno.h> #incluir <pthread.h> #include <semáforo.h> #incluir <stdbool.h> #incluir <stdio.h> #incluir <stdlib.h> #ifndef __STDC_LIB_EXT1__ typedef int errno_t ; #terminara si enumeración { EOK = 0 , }; // Implementación mutex simplificada estructura guard_t { sem_t sem_guardia ; }; typedef struct guard_t guard_t ; // Inicializar el mutex simplificado errno_t guard_init ( guard_t * guard , bool between_processes ) { int r ; r = sem_init ( & guard -> sem_guard , between_processes , 1 ); si ( r == -1 ) { devuelve errno ; } devolver EOK ; } // Liberar el mutex simplificado void guard_free ( guard_t * guard ) { sem_destroy ( & guardia -> sem_guard ); } // Entrando en la sección crítica errno_t guardia_enter ( guardia_t * guardia ) { int r ; hacer { r = sem_esperar ( & guardia -> sem_guardia ); } while (( r == -1 ) && ( errno == EINTR )); si ( r == -1 ) { devuelve errno ; } devolver EOK ; } // Salir de la sección crítica errno_t guard_leave ( guard_t * guard ) { int r ; r = sem_post ( & guard -> sem_guard ); si ( r == -1 ) { devuelve errno ; } devolver EOK ; } // Contador protegido por un mutex simplificado estructura contador_seguro_t { guardia_t bloquear ; contador int ; }; enumeración { // Número de operaciones de aumento/decremento OPERACIONES_CUENTA = 100000 , }; // Subproceso incrementando el contador vacío * subproceso_inc_func ( vacío * subproceso_datos ) { struct contador_seguro_t * contador_seguro = hilo_datos ; for ( int i = 0 ; i < OPERATIONS_COUNT ; ++ i ) { guard_enter ( & contador_seguro -> bloquear ); ++ contador_seguro -> contador ; guard_leave ( & contador_seguro -> bloquear ); } } // Hilo decrementando el contador vacío * thread_dec_func ( vacío * thread_data ) { struct contador_seguro_t * contador_seguro = hilo_datos ; for ( int i = 0 ; i < OPERATIONS_COUNT ; ++ i ) { guard_enter ( & contador_seguro -> bloquear ); -- contador_seguro -> contador ; guard_leave ( & contador_seguro -> bloquear ); } } // Muestra un mensaje de error de acuerdo con su código void print_error ( errno_t errnum , const char * error_text ) { número de error = número de error ; error ( texto_error ); } int principal ( int argc , char ** argv ) { errno_t errnum ; // inicialización estructura contador_seguro_t contador_seguro ; contador_seguro . contador = 0 ; guardia_t bloquear ; errnum = guard_init ( & safe_counter . lock , false ); si ( número de error ) { print_error ( errnum , "Error al inicializar el bloqueo mutex" ); salir ( EXIT_FAILURE ); } // Iniciar dos hilos pthread_t hilo_inc ; errnum = pthread_create ( & thread_inc , NULL , thread_inc_func , & safe_counter ); si ( número de error ) { print_error ( errnum , "Error al crear thread_inc" ); salir ( EXIT_FAILURE ); } pthread_t thread_dec ; errnum = pthread_create ( & thread_dec , NULL , thread_dec_func , & safe_counter ); si ( número de error ) { print_error ( errnum , "Error al crear thread_dec" ); salir ( EXIT_FAILURE ); } // Espera a que los hilos terminen de ejecutarse errnum = pthread_join ( thread_inc , NULL ); si ( número de error ) { print_error ( errnum , "Error esperando thread_inc" ); salir ( EXIT_FAILURE ); } errnum = pthread_join ( thread_dec , NULL ); si ( número de error ) { print_error ( errnum , "Error esperando thread_dec" ); salir ( EXIT_FAILURE ); } // Liberar datos guardia_libre ( & bloqueo ); // Muestra el resultado de los hilos, "0" printf ( "Contador: %d \n " , contador_seguro . contador ); devuelve SALIR_ÉXITO ; }

Ejemplo de sincronización de búfer de anillo

Sincronizar el búfer circular es un poco más complicado que proteger la sección crítica: ya hay dos semáforos y se les agregan variables adicionales . El ejemplo muestra la estructura y las funciones básicas necesarias para sincronizar un búfer de anillo C utilizando la interfaz POSIX . Esta implementación permite que un subproceso escriba datos en el búfer circular de forma cíclica y otro subproceso los lea de forma asíncrona.

Ejemplo de implementación de una primitiva de sincronización para un búfer circular usando semáforos en C (POSIX) #include <errno.h> #include <semáforo.h> #incluir <stdio.h> #ifndef __STDC_LIB_EXT1__ typedef int errno_t ; #endif enumeración { EOK = 0 , }; estructura anillo_buffer_t { tamaño_t longitud ; tamaño_t w_índice ; tamaño_t r_index ; sem_t sem_r ; sem_t sem_w ; }; errno_t ring_buffer_init ( estructura ring_buffer_t * rbuf , tamaño_t longitud ) { rbuf -> longitud = longitud ; rbuf -> r_index = 0 ; rbuf -> índice_w = 0 ; int r ; r = sem_init ( & rbuf -> sem_r , 1 , 0 ); si ( r == -1 ) { devuelve errno ; } errno_t errnum ; r = sem_init ( & rbuf -> sem_w , 1 , longitud ); si ( r == -1 ) { errnum = errno ; ir a abortar_sem_r ; } devolver EOK ; abortando_sem_r : sem_destroy ( & rbuf -> sem_r ); devolver número de error ; } anular ring_buffer_free ( estructura ring_buffer_t * rbuf ) { sem_destroy ( & rbuf -> sem_w ); sem_destroy ( & rbuf -> sem_r ); } errno_t ring_buffer_write_begin ( estructura ring_buffer_t * rbuf ) { int r ; hacer { r = sem_esperar ( & rbuf -> sem_w ); } while (( r == -1 ) && ( errno == EINTR )); si ( r == -1 ) { devuelve errno ; } devolver EOK ; } errno_t ring_buffer_write_end ( estructura ring_buffer_t * rbuf ) { ++ rbuf- > w_index ; if ( rbuf -> w_index >= rbuf -> longitud ) { rbuf -> índice_w = 0 ; } int r ; r = sem_post ( & rbuf -> sem_r ); si ( r == -1 ) { devuelve errno ; } devolver EOK ; } errno_t ring_buffer_read_begin ( estructura ring_buffer_t * rbuf ) { int r ; hacer { r = sem_esperar ( & rbuf -> sem_r ); } while (( r == -1 ) && ( errno == EINTR )); si ( r == -1 ) { devuelve errno ; } devolver EOK ; } errno_t ring_buffer_read_end ( estructura ring_buffer_t * rbuf ) { ++ rbuf -> r_index ; if ( rbuf -> r_index >= rbuf -> longitud ) { rbuf -> r_index = 0 ; } int r ; r = sem_post ( & rbuf -> sem_w ); si ( r == -1 ) { devuelve errno ; } devolver EOK ; }

Detalles de implementación

Sobre los sistemas operativos

En general, los sistemas operativos realizan lecturas y escrituras atómicas del valor del contador del semáforo, pero los detalles de implementación pueden variar en diferentes arquitecturas. Al adquirir un semáforo, el sistema operativo debe disminuir atómicamente el valor del contador, después de lo cual el proceso puede continuar su trabajo. Si, como resultado de decrementar el contador, el valor puede volverse negativo, entonces el sistema operativo debe suspender la ejecución del proceso hasta que el valor del contador sea tal que la operación de decremento conduzca a un resultado no negativo [16] . En este caso, dependiendo de la arquitectura a nivel de implementación, se puede realizar tanto un intento de reducción del valor del semáforo [67] como su disminución con resultado negativo [68] . En el nivel de la interfaz de la aplicación, se suele suponer que el valor mínimo de un semáforo es 0 [3] . Cuando aumenta el valor del semáforo en el que se bloquearon los procesos, se desbloquea el siguiente proceso y el valor del semáforo a nivel de aplicación permanece igual a cero.

Un bloqueo a nivel del sistema operativo generalmente no implica una espera física en el procesador, sino que transfiere el control del procesador a otra tarea, mientras que un semáforo en espera de liberación ingresa a la cola de tareas bloqueadas por este semáforo [69] . Si la cantidad de tareas listas para la ejecución es menor que la cantidad de procesadores, el kernel del sistema operativo puede cambiar los procesadores libres al modo de ahorro de energía antes de que ocurra cualquier evento.

A nivel de procesador

En arquitecturas x86 y x86_64

Para sincronizar el trabajo de los procesadores en sistemas multiprocesador, existen instrucciones especiales que le permiten proteger el acceso a cualquier celda. En la arquitectura x86 , Intel proporciona un prefijo para una serie de instrucciones de procesador LOCKque le permiten realizar operaciones atómicas en celdas de memoria. Las operaciones de celda realizadas con el prefijo LOCKbloquean el acceso de otros procesadores a la celda, lo que en un nivel primitivo permite organizar semáforos ligeros con un bucle de espera activo [70] .

La disminución atómica de un valor de semáforo en 1 se puede hacer con una instrucción DECLcon el prefijo LOCK, que establece el indicador de signo CSsi el valor resultante es menor que cero. Una característica de este enfoque es que el valor del semáforo puede ser menor que cero, por lo que después de disminuir el contador, la bandera CSse puede verificar mediante la instrucción JNSy, si el signo es negativo, el sistema operativo puede bloquear la tarea actual [71] .

La instrucción se puede utilizar para incrementar el valor de un semáforo en 1 atómicamente LOCK INCL. Si el valor resultante es negativo o igual a cero, significa que hay tareas pendientes, en cuyo caso el sistema operativo puede desbloquear la siguiente tarea. Para saltarse procesos de desbloqueo se puede utilizar la instrucción JG, que salta a la etiqueta si los flags de resultado de operación cero ( ZF) y signo de resultado ( SF) se ponen a 0, es decir, si el valor es mayor que 0 [72] .

Durante el bloqueo, en los casos en que no hay tareas actuales, se puede usar una instrucción para poner el HLTprocesador en un modo de bajo consumo mientras espera las interrupciones [73] , que primero deben habilitarse con la instrucción STI. Sin embargo, en los procesadores modernos, puede ser más óptimo usar las instrucciones MWAITy MONITOR. La instrucción MWAITes similar HLT, pero le permite activar el procesador escribiendo en una celda de memoria en la dirección especificada en MONITOR. NWAITse puede usar para monitorear los cambios de la ranura del semáforo, sin embargo, en los sistemas operativos multitarea, esta instrucción se usa para monitorear un indicador para ejecutar el programador de tareas en un kernel determinado [74] .

Se puede reducir el consumo de energía durante el ciclo de reposo activo mediante la instrucción PAUSE[75] .

En la arquitectura ARM

La arquitectura ARMv7 utiliza los llamados monitores exclusivos locales y globales para sincronizar la memoria entre los procesadores, que son máquinas de estado que controlan el acceso atómico a las celdas de memoria [76] [77] . Se puede realizar una lectura atómica de una celda de memoria mediante la instrucción LDREX[78] y una escritura atómica mediante la instrucción STREX, que también devuelve el indicador de éxito de la operación [79] .

Para disminuir el valor de un semáforo, debe esperar hasta que su contador sea mayor que cero. La espera se puede implementar de diferentes maneras:

  • un bucle de espera activo en el caso de un semáforo ligero, que comprueba periódicamente el valor del contador [80] mediante la instrucción LDREX;
  • bloqueo con la transferencia del procesador a un modo de suspensión de ahorro de energía usando instrucciones de espera WFIde interrupción o esperar un evento WFE[81] [82] ;
  • cambio de contexto para ejecutar otra tarea en lugar de bloquear el procesador [83] .

A nivel de un sistema operativo multitarea, se puede utilizar una combinación de estos métodos para proporcionar la máxima utilización del procesador con una transición al modo de ahorro de energía durante los tiempos de inactividad.

Incrementar el valor de un semáforo puede ser una lectura cíclica del valor actual del contador a través de la instrucción LDREX, luego incrementar una copia del valor e intentar volver a escribir en la ubicación del contador usando la instrucción STREX[84] . Después de un registro exitoso del contador, si su valor inicial era cero, se requiere reanudar la ejecución de las tareas bloqueadas [84] , que en el caso de un cambio de contexto se puede solucionar mediante los sistemas operativos [80] . Si el procesador se bloqueó con la instrucción WFE, se puede desbloquear con la instrucción SEVque notifica la presencia de cualquier evento [85] .

Luego de decrementar o incrementar el valor del semáforo, se ejecuta la instrucción para DMBasegurar la integridad de la memoria del recurso protegido por el semáforo [86] .

Véase también

Notas

Documentación

  1. Función sem_init() Archivado el 2 de mayo de 2019 en Wayback Machine .
  2. Función sem_destroy() Archivado el 2 de mayo de 2019 en Wayback Machine .
  3. Función sem_open() Archivado el 2 de mayo de 2019 en Wayback Machine .
  4. Función sem_close() Archivado el 2 de mayo de 2019 en Wayback Machine .
  5. Función sem_unlink() Archivado el 2 de mayo de 2019 en Wayback Machine .
  6. Función sem_wait() Archivado el 2 de mayo de 2019 en Wayback Machine .
  7. Función sem_timedwait() Archivado el 2 de mayo de 2019 en Wayback Machine .
  8. Función sem_trywait() Archivado el 29 de junio de 2019 en Wayback Machine .
  9. Función sem_post() Archivado el 2 de mayo de 2019 en Wayback Machine .
  10. Función sem_getvalue() Archivado el 2 de mayo de 2019 en Wayback Machine .
  11. Función semget() Archivado el 17 de junio de 2019 en Wayback Machine .
  12. Función semop() Archivado el 25 de junio de 2019 en Wayback Machine .
  13. Función semctl() Archivado el 20 de junio de 2019 en Wayback Machine .
  14. función eventfd() Archivado el 8 de junio de 2019 en Wayback Machine .
  15. Función CreateSemaphoreA() Archivado el 2 de mayo de 2019 en Wayback Machine .
  16. Función OpenSemaphoreW() Archivado el 2 de mayo de 2019 en Wayback Machine .
  17. Función CloseHandle() Archivado el 2 de mayo de 2019 en Wayback Machine .
  18. Función WaitForSingleObject() Archivado el 2 de mayo de 2019 en Wayback Machine .
  19. Función WaitForMultipleObjects() Archivado el 2 de mayo de 2019 en Wayback Machine .
  20. Función ReleaseSemaphore() Archivado el 2 de mayo de 2019 en Wayback Machine .
  21. Lenguaje C, sem_t Archivado el 5 de mayo de 2019 en Wayback Machine .
  22. Tongue of Hell, GNAT.Semaphores Archivado el 26 de mayo de 2019 en Wayback Machine .
  23. Lenguaje C++, boost::interprocess::interprocess_semaphore Archivado el 3 de mayo de 2019 en Wayback Machine .
  24. Lenguaje C#, System.Threading Archivado el 30 de octubre de 2020 en Wayback Machine .
  25. Lenguaje C#, Semáforo
  26. Lenguaje D, core.sync.semaphore Archivado el 3 de mayo de 2019 en Wayback Machine .
  27. Idioma D, semáforo Archivado el 3 de mayo de 2019 en Wayback Machine .
  28. The Go Language, golang.org/x/sync/semaphore Archivado el 3 de mayo de 2019 en Wayback Machine .
  29. Lenguaje Java,java.util.concurrent
  30. Lenguaje Java,java.util.concurrent.Semaphore
  31. Python language, threading Archivado el 25 de enero de 2022 en Wayback Machine .
  32. Python Language, asyncio Archivado el 5 de mayo de 2019 en Wayback Machine .
  33. Python language, threading.Semaphore Archivado el 25 de enero de 2022 en Wayback Machine .
  34. lenguaje Python, asyncio.Semaphore Archivado el 7 de abril de 2019 en Wayback Machine .

Fuentes

  1. ↑ 1 2 El grupo abierto. 4. Conceptos generales: 4.17 Semáforo  // Especificaciones básicas de Open Group Número 7. - pubs.opengroup.org. — Fecha de acceso: 12/06/2019.
  2. Ching-Kuang Shene. Semáforos, Concepto Básico  : [ arch. 15/06/2020 ] // Programación multiproceso con ThreadMentor: un tutorial. — Universidad Tecnológica de Michigan. — Fecha de acceso: 07/06/2019.
  3. 1 2 3 4 5 6 7 8 9 10 Cameron Hughes, Tracey Hughes. Programación Paralela y Distribuida con C++ . — Editorial Williams. - pág. 194. - 667 pág. — ISBN 9785845906861 .
  4. 1 2 Tanenbaum, 2011 , 2.3.5. Semáforos, pág. 164.
  5. ↑ 1 2 pthread_mutex_unlock(3): bloquear/desbloquear mutex -  página del manual de Linux . linux.die.net. Consultado el 1 de mayo de 2019. Archivado desde el original el 1 de mayo de 2019.
  6. 1 2 Allen B. Downey, 2016 , 3.1 Señalización, pág. 11-12.
  7. 1 2 Allen B. Downey, 2016 , 3.6.4 Solución de barrera, p. 29
  8. ↑ 1 2 3 4 Andrew S. Tanenbaum, T. Austin. Arquitectura informática  = Organización informática estructurada. — 5ª edición. - San Petersburgo: Piter, 2010. - S. 510-516. — 844 pág. — ISBN 9785469012740 .
  9. 1 2 Allen B. Downey, 2016 , 3.6 Barrera, p. 21-22.
  10. 1 2 Allen B. Downey, 2016 , 4.2 Problema de lectores y escritores, p. 65-66.
  11. 1 2 Tanenbaum, 2011 , 6.2. Introducción a los interbloqueos, pág. 511.
  12. 1 2 3 Allen B. Downey, 2016 , 4.2.3 Inanición, pág. 71.
  13. ↑ 1 2 sem_wait  // La especificación única de UNIX®, versión 2. - pubs.opengroup.org, 1997. - 24 de octubre. — Fecha de acceso: 09/06/2019.
  14. ↑ 1 2 Inversión de prioridad - herencia de prioridad  // Wiki de The Linux Foundation. wiki.linuxfoundation.org. — Fecha de acceso: 09/06/2019.
  15. Tanenbaum, 2011 , 2.3.5. Semáforos, pág. 162.
  16. 1 2 3 Tanenbaum, 2011 , 2.3.5. Semáforos, pág. 162-163.
  17. 1 2 3 Allen B. Downey, 2016 , 2.1 Definición, pág. 7-8.
  18. Tanenbaum, 2011 , 2.3.5. Semáforos, pág. 163.
  19. ↑ 1 2 Pobegailo AP . Programación del sistema en Windows . - San Petersburgo: BHV-Petersburg, 2006. - S. 137–142. — 1056 pág. — ISBN 9785941577927 .
  20. ↑ 1 2 Referencia de la API de Java . docs.oracle.com. — Fecha de acceso: 04/05/2019.
  21. Oleg Tsilyurik. Herramientas de programación del kernel: Parte 73. Paralelismo y sincronización. Cerraduras. parte 1 - www.ibm.com, 2013. - 13 de agosto. — Fecha de acceso: 04/05/2019.
  22. Bovet, Cesati, 2002 , p. 24
  23. Tanenbaum, 2011 , 2.3.7. monitores, pág. 176.
  24. ↑ 1 2 3 Remi Denis-Courmont. Otros usos de futex  // Remlab. - Remlab.net, 2016. - 21 de septiembre. — Fecha de acceso: 15/06/2019.
  25. Ching-Kuang Shene. Bloqueos de exclusión mutua: mutex, Concepto básico  : [ arch. 15/06/2020 ] // Programación multiproceso con ThreadMentor: un tutorial. — Universidad Tecnológica de Michigan. — Fecha de acceso: 07/06/2019.
  26. Uso de mutexes  // AIX 7.2, Programación para AIX. — Centro de conocimientos de IBM. — Fecha de acceso: 15/06/2020.
  27. Tanenbaum, 2011 , 2.3.5. Semáforos, Resolviendo el Problema del Productor y el Consumidor, p. 164.
  28. Tanenbaum, 2011 , 2.3.6. Mutexes, pág. 165.
  29. ↑ 1 2 Ching-Kuang Shene. Tres técnicas comúnmente utilizadas  // Programación multiproceso con ThreadMentor: un tutorial. — Universidad Tecnológica de Michigan. — Fecha de acceso: 07/06/2019.
  30. ↑ 1 2 El grupo abierto. pthread_mutexattr_setprotocol  // La especificación única de UNIX®, versión 2. - pubs.opengroup.org, 1997. - 24 de octubre. — Fecha de acceso: 09/06/2019.
  31. Tanenbaum, 2011 , 2.3.7. monitores, pág. 170-176.
  32. 1 2 3 Allen B. Downey, 2016 , 3.7.6 Torniquete precargado, pág. 43.
  33. Allen B. Downey, 2016 , 3.5.4 Solución de barrera, p. 29
  34. 1 2 3 Allen B. Downey, 2016 , 4.2.5 Solución para lectores y escritores sin hambre, p. 75.
  35. 1 2 3 Allen B. Downey, 2016 , 4.2.2 Solución de lectores y escritores, p. 69-71.
  36. C.-K. Shene. ThreadMentor: el problema del productor/consumidor (o del búfer acotado)  // Programación multiproceso con ThreadMentor: un tutorial. — Universidad Tecnológica de Michigan. — Fecha de acceso: 01/07/2019.
  37. Allen B. Downey, 2016 , 4.1.2 Solución productor-consumidor, p. 59-60.
  38. Allen B. Downey, 2016 , 3.3.2 Solución de encuentro, p. quince.
  39. Allen B. Downey, 2016 , 3.7.5 Solución de barrera reutilizable, pág. 41-42.
  40. Remi Denis-Courmont. Variable de condición con futex  // Remlab. - Remlab.net, 2016. - 21 de septiembre. — Fecha de acceso: 16/06/2019.
  41. ↑ 123 Microsoft . _ _ Función ReleaseSemaphore (synchapi.h) . docs.microsoft.com. — Fecha de acceso: 05/05/2019.
  42. ↑ 1 2 3 4 Andrew D. Birrell. Implementación de variables de condición con semáforos  // Microsoft Research. - www.microsoft.com, 2003. - 1 de enero.
  43. Oleg Tsilyurik. Herramientas de programación del kernel: Parte 73. Paralelismo y sincronización. Cerraduras. parte 1 - www.ibm.com, 2013. - 13 de agosto. — Fecha de acceso: 12/06/2019.
  44. 1 2 3 4 5 Allen B. Downey, 2016 , 4.4 Filósofos de la cena, p. 87-88.
  45. 1 2 Allen B. Downey, 2016 , 5.8 El problema de la montaña rusa, p. 153.
  46. ↑ 1 2 eventfd(2) - Página del manual de Linux . man7.org. — Fecha de acceso: 08/06/2019.
  47. ↑ 1 2 semop  // The Open Group Base Specifications Edición 7. - pubs.opengroup.org. — Fecha de acceso: 12/06/2019.
  48. IEEE, El grupo abierto. sem_trywait  // The Open Group Base Specifications Número 7. - pubs.opengroup.org, 2008. - 24 de octubre. — Fecha de acceso: 29/06/2019.
  49. 1 2 3 4 5 6 7 8 Allen B. Downey, 2016 , 4.3 Mutex sin hambre, p. 81-82.
  50. Tanenbaum, 2011 , 6.1.2. Conseguir un recurso, pág. 510.
  51. Rohit Chandra, Leo Dagum, David Kohr, Ramesh Menon, Dror Maydan.  Programación paralela en OpenMP ] . - Morgan Kaufmann, 2001. - Pág. 151. - 250 p. — ISBN 9781558606715 .
  52. Tanenbaum, 2011 , 6.1.2. Conseguir un recurso, pág. 510–511.
  53. Tanenbaum, 2011 , 6.1.2. Conseguir un recurso, pág. 511.
  54. ↑ 1 2 Víctor Yodaiken. Contra herencia de prioridad  // Contra herencia de prioridad. - Finite State Machine Labs, 2004. - 23 de septiembre.
  55. ↑ 1 2 IEEE, El grupo abierto. semaphore.h - semáforos  // The Open Group Base Specifications Edición 7, edición de 2018. — pubs.opengroup.org. — Fecha de acceso: 08/06/2019.
  56. sem_timedwait.3p - Página del manual de Linux . man7.org. — Fecha de acceso: 05/05/2019.
  57. 112521 - monotónico sem_timedwait . — bugzilla.kernel.org. — Fecha de acceso: 05/05/2019.
  58. sem_timedwait(), sem_timedwait_monotonic()  // Sistema operativo en tiempo real QNX Neutrino. — www.qnx.com. — Fecha de acceso: 05/05/2019.
  59. futex(2) - Página del manual de Linux . man7.org. — Fecha de acceso: 23/06/2019. (Sección "NOTAS".)
  60. El grupo abierto. 2. Información general: 2.7 XSI Interprocess Communication  // The Open Group Base Specifications Issue 7. - pubs.opengroup.org. — Fecha de acceso: 11/06/2019.
  61. Vikram Shukla. Semáforos en Linux  (inglés) (2007-24-05). — El artículo original está en web.archive.org, pero está incompleto. Consultado el 12 de junio de 2019. Archivado desde el original el 12 de junio de 2020.
  62. ↑ 1 2 semget  // Especificaciones básicas de Open Group Número 7. - pubs.opengroup.org. — Fecha de acceso: 12/06/2019.
  63. semctl  // Especificaciones básicas de Open Group Número 7. - pubs.opengroup.org. — Fecha de acceso: 12/06/2019.
  64. Microsoft . Función WaitForMultipleObjects (synchapi.h) . docs.microsoft.com. — Fecha de acceso: 05/05/2019.
  65. M. Ben-Ari, Môtî Ben-Arî. Principios de Programación Concurrente y Distribuida  : [ ing. ] . - Addison-Wesley, 2006. - P. 132. - 388 p. — ISBN 9780321312839 .
  66. Semáforos - Go Language Patterns . - www.golangpatterns.info — Fecha de acceso: 08/06/2019.
  67. ARM, 2009 , 1.3.3 Implementando un semáforo, p. 14-15.
  68. Bovet, Cesati, 2002 , Semáforos: Obtener y liberar semáforos, p. 175.
  69. Bovet, Cesati, 2002 , Sincronización y regiones críticas: semáforos, p. 24-25.
  70. Ruslán Ablyazov. Programación en lenguaje ensamblador en la plataforma x86-64 . - Litros, 2017. - S. 273-275. — 304 pág. — ISBN 9785040349203 .
  71. Bovet, Cesati, 2002 , Obtener y liberar semáforos, p. 175.
  72. Bovet, Cesati, 2002 , Obtener y liberar semáforos, p. 174.
  73. Linux BootPrompt-HowTo: Argumentos generales de arranque no específicos del dispositivo . — www.tldp.org. — Fecha de acceso: 03/05/2019.
  74. Corey Gough, Ian Steiner, Winston Saunders. Servidores energéticamente eficientes: Blueprints para la optimización del centro de datos . - Apress, 2015. - P. 175. - 347 p. — ISBN 9781430266389 .
  75. Bovet, Cesati, 2002 , p. 169-170.
  76. ARM, 2009 , 1.2.1 LDREX y STREX, p. cuatro
  77. ARM, 2009 , 1.2.2 Monitores exclusivos, p. 5.
  78. ARM, 2009 , 1.2.1 LDREX y STREX, LDREX, p. cuatro
  79. ARM, 2009 , 1.2.1 LDREX y STREX, STREX, p. cuatro
  80. 1 2 ARM, 2009 , 1.3.1 Funciones de ahorro de energía, p. 9.
  81. ARM, 2009 , 1.3.3 Implementando un semáforo, p. 14: "WAIT_FOR_UPDATE y SIGNAL_UPDATE se describen en Funciones de ahorro de energía en la página 1-9".
  82. ARM, 2009 , 1.3.1 Funciones de ahorro de energía, p. 9-12.
  83. ARM, 2009 , 1.3.1 Funciones de ahorro de energía: reprogramación como función de ahorro de energía, p. once.
  84. 1 2 ARM, 2009 , 1.3.3 Implementando un semáforo, p. catorce.
  85. ARM, 2009 , 1.3.1 Funciones de ahorro de energía, p. 10-11.
  86. ARM, 2009 , 1.2.3 Barreras de memoria, p. ocho.

Literatura

  • Allen B. Downey. El librito de los semáforos  : [ ing. ]  : [ arq. 20 de mayo de 2020 ]. — Segunda Edición, Versión 2.2.1. - 2016. - 279 págs.
  • Andrew S. Tanenbaum. Sistemas operativos  modernos = Sistemas operativos modernos. — 3ra edición. - San Petersburgo: Peter: Editorial "Peter", 2011. - S. 162-163. — 1117 pág. — (Clásicos de la informática). — ISBN 9785459007572 .
  • Daniel Pierre Bovet, Marco Cesari. Entendiendo el Kernel de Linux . - O'Reilly Media, Inc., 2002. - P. 173-177. — 786 pág. — ISBN 9780596002138 .
  • BRAZO. Primitivas de sincronización ARM  : Artículo de desarrollo: [ ing. ]  : [ arq. 20 de mayo de 2020 ]. - 2009. - 19 de agosto. — 28 s.