stringtranslate.com

Gestión de la memoria

La gestión de memoria (también gestión de memoria dinámica , asignación de almacenamiento dinámico o asignación de memoria dinámica ) es una forma de gestión de recursos aplicada a la memoria de la computadora . El requisito esencial de la gestión de memoria es proporcionar formas de asignar dinámicamente porciones de memoria a los programas cuando estos lo soliciten y liberarla para su reutilización cuando ya no sea necesaria. Esto es fundamental para cualquier sistema informático avanzado en el que pueda estar en marcha más de un proceso en cualquier momento. [1]

Se han ideado varios métodos que aumentan la eficacia de la gestión de la memoria. Los sistemas de memoria virtual separan las direcciones de memoria utilizadas por un proceso de las direcciones físicas reales, lo que permite la separación de procesos y aumenta el tamaño del espacio de direcciones virtuales más allá de la cantidad disponible de RAM mediante paginación o intercambio a almacenamiento secundario . La calidad del administrador de memoria virtual puede tener un efecto importante en el rendimiento general del sistema . El sistema permite que una computadora parezca que tiene más memoria disponible que la que está físicamente presente, lo que permite que varios procesos la compartan.

En algunos sistemas operativos , por ejemplo OS/360 y sucesores , [2] la memoria es administrada por el sistema operativo. [nota 1] En otros sistemas operativos, por ejemplo los sistemas operativos tipo Unix , la memoria es administrada a nivel de aplicación.

La gestión de memoria dentro de un espacio de direcciones generalmente se clasifica como gestión de memoria manual o gestión de memoria automática.

Gestión manual de memoria

Un ejemplo de fragmentación externa

La tarea de satisfacer una solicitud de asignación consiste en localizar un bloque de memoria no utilizada de tamaño suficiente. Las solicitudes de memoria se satisfacen asignando porciones de un gran conjunto [nota 2] de memoria llamado montón [nota 3] o almacén libre . En un momento dado, algunas partes del montón están en uso, mientras que otras están "libres" (sin usar) y, por lo tanto, disponibles para futuras asignaciones. En el lenguaje C, se llama a la función que asigna memoria del montón mallocy se llama a la función que toma la memoria previamente asignada y la marca como "libre" (para ser utilizada en futuras asignaciones) free. [nota 4]

Varios problemas complican la implementación, como la fragmentación externa , que surge cuando hay muchos espacios pequeños entre los bloques de memoria asignados, lo que invalida su uso para una solicitud de asignación. Los metadatos del asignador también pueden inflar el tamaño de asignaciones (individualmente) pequeñas. Esto se suele gestionar mediante la fragmentación . El sistema de gestión de memoria debe realizar un seguimiento de las asignaciones pendientes para garantizar que no se superpongan y que nunca se "pierda" memoria (es decir, que no haya " fugas de memoria ").

Eficiencia

El algoritmo específico de asignación de memoria dinámica implementado puede afectar significativamente el rendimiento. Un estudio realizado en 1994 por Digital Equipment Corporation ilustra los costos operativos que implican una variedad de asignadores. La longitud de ruta de instrucciones promedio más baja requerida para asignar una sola ranura de memoria fue 52 (medida con un perfilador de nivel de instrucción en una variedad de software). [1]

Implementaciones

Dado que la ubicación precisa de la asignación no se conoce de antemano, se accede a la memoria de forma indirecta, normalmente a través de una referencia de puntero . El algoritmo específico utilizado para organizar el área de memoria y asignar y desasignar fragmentos está interconectado con el núcleo y puede utilizar cualquiera de los siguientes métodos:

Asignación de bloques de tamaño fijo

La asignación de bloques de tamaño fijo, también llamada asignación de grupo de memoria, utiliza una lista libre de bloques de memoria de tamaño fijo (a menudo todos del mismo tamaño). Esto funciona bien para sistemas integrados simples en los que no es necesario asignar objetos grandes, pero sufre fragmentación , especialmente con direcciones de memoria largas. Sin embargo, debido a la sobrecarga significativamente reducida, este método puede mejorar sustancialmente el rendimiento de los objetos que necesitan asignación y desasignación frecuentes, por lo que se utiliza a menudo en los videojuegos .

Bloques de amigos

En este sistema, la memoria se asigna en varios grupos de memoria en lugar de solo uno, donde cada grupo representa bloques de memoria de un tamaño de una determinada potencia de dos , o bloques de alguna otra progresión de tamaño conveniente. Todos los bloques de un tamaño particular se guardan en una lista enlazada ordenada o árbol y todos los bloques nuevos que se forman durante la asignación se agregan a sus respectivos grupos de memoria para su uso posterior. Si se solicita un tamaño menor que el disponible, se selecciona el tamaño más pequeño disponible y se divide. Se selecciona una de las partes resultantes y el proceso se repite hasta que se completa la solicitud. Cuando se asigna un bloque, el asignador comenzará con el bloque más pequeño lo suficientemente grande como para evitar romper bloques innecesariamente. Cuando se libera un bloque, se compara con su compañero. Si ambos están libres, se combinan y se colocan en la lista de bloques amigos de tamaño correspondientemente mayor.

Asignación de losas

Este mecanismo de asignación de memoria preasigna fragmentos de memoria adecuados para albergar objetos de un determinado tipo o tamaño. [4] Estos fragmentos se denominan cachés y el asignador solo tiene que realizar un seguimiento de una lista de ranuras de caché libres. La construcción de un objeto utilizará cualquiera de las ranuras de caché libres y la destrucción de un objeto agregará una ranura nuevamente a la lista de ranuras de caché libres. Esta técnica alivia la fragmentación de la memoria y es eficiente ya que no hay necesidad de buscar una porción adecuada de memoria, ya que cualquier ranura abierta será suficiente.

Asignación de pila

Muchos sistemas similares a Unix, así como Microsoft Windows, implementan una función llamada allocapara asignar dinámicamente memoria de pila de una manera similar a la basada en heap malloc. Un compilador normalmente la traduce a instrucciones en línea que manipulan el puntero de pila. [5] Aunque no hay necesidad de liberar manualmente la memoria asignada de esta manera, ya que se libera automáticamente cuando la función que se llamó allocaretorna, existe un riesgo de desbordamiento. Y dado que alloca es una expansión ad hoc vista en muchos sistemas pero nunca en POSIX o el estándar C, su comportamiento en caso de un desbordamiento de pila no está definido.

Existe una versión más segura de alloca llamada _malloca, que informa errores, en Microsoft Windows. Requiere el uso de _freea. [6] gnulib proporciona una interfaz equivalente, aunque en lugar de lanzar una excepción SEH en caso de desbordamiento, delega a malloc cuando se detecta un tamaño demasiado grande. [7] Se puede emular una característica similar utilizando contabilidad manual y comprobación de tamaño, como en los usos de alloca_accounten glibc. [8]

Gestión automatizada de memoria

La gestión adecuada de la memoria en una aplicación es un problema difícil y se han ideado varias estrategias diferentes para gestionar la memoria.

Gestión automática de variables de la pila de llamadas

En muchas implementaciones de lenguajes de programación, el entorno de ejecución del programa asigna automáticamente memoria en la pila de llamadas para las variables locales no estáticas de una subrutina , llamadas variables automáticas , cuando se llama a la subrutina, y libera automáticamente esa memoria cuando se sale de la subrutina. Las declaraciones especiales pueden permitir que las variables locales conserven valores entre invocaciones del procedimiento, o pueden permitir que otras subrutinas accedan a las variables locales. La asignación automática de variables locales hace posible la recursión , hasta una profundidad limitada por la memoria disponible.

Recolección de basura

La recolección de basura es una estrategia para detectar automáticamente la memoria asignada a objetos que ya no se pueden utilizar en un programa y devolver esa memoria asignada a un grupo de ubicaciones de memoria libres. Este método contrasta con la administración de memoria "manual", en la que un programador codifica explícitamente las solicitudes de memoria y las liberaciones de memoria en el programa. Si bien la recolección de basura automática tiene las ventajas de reducir la carga de trabajo del programador y evitar ciertos tipos de errores de asignación de memoria, la recolección de basura requiere recursos de memoria propios y puede competir con el programa de aplicación por el tiempo del procesador.

Recuento de referencias

El conteo de referencias es una estrategia para detectar que un programa ya no puede utilizar la memoria, manteniendo un contador que indica cuántos punteros independientes apuntan a la memoria. Siempre que un nuevo puntero apunta a una parte de la memoria, el programador debe aumentar el contador. Cuando el puntero cambia de lugar, o cuando ya no apunta a ninguna zona o se ha liberado, el contador debe disminuir. Cuando el contador llega a cero, la memoria debe considerarse no utilizada y liberada. Algunos sistemas de conteo de referencias requieren la participación del programador y otros son implementados automáticamente por el compilador. Una desventaja del conteo de referencias es que pueden desarrollarse referencias circulares que provoquen una fuga de memoria. Esto se puede mitigar añadiendo el concepto de "referencia débil" (una referencia que no participa en el conteo de referencias, pero que recibe una notificación cuando la zona a la que apunta ya no es válida) o combinando el conteo de referencias y la recolección de basura.

Reservas de memoria

Un grupo de memoria es una técnica de desasignación automática de memoria en función del estado de la aplicación, como el ciclo de vida de una solicitud o transacción. La idea es que muchas aplicaciones ejecuten grandes fragmentos de código que pueden generar asignaciones de memoria, pero que haya un punto en la ejecución en el que se sepa que todos esos fragmentos ya no son válidos. Por ejemplo, en un servicio web, después de cada solicitud, el servicio web ya no necesita ninguna de las memorias asignadas durante la ejecución de la solicitud. Por lo tanto, en lugar de realizar un seguimiento de si se hace referencia a la memoria o no, la memoria se asigna de acuerdo con la solicitud o la etapa del ciclo de vida con la que está asociada. Cuando esa solicitud o etapa ha pasado, toda la memoria asociada se desasigna simultáneamente.

Sistemas con memoria virtual

La memoria virtual es un método para disociar la organización de la memoria del hardware físico. Las aplicaciones operan en la memoria a través de direcciones virtuales . Cada intento de la aplicación de acceder a una dirección de memoria virtual en particular da como resultado que la dirección de memoria virtual se traduzca a una dirección física real . [9] De esta manera, la adición de memoria virtual permite un control granular sobre los sistemas de memoria y los métodos de acceso.

En los sistemas de memoria virtual, el sistema operativo limita el acceso de un proceso a la memoria. Esta función, denominada protección de memoria , se puede utilizar para impedir que un proceso lea o escriba en la memoria que no le está asignada, lo que evita que el código malicioso o que funcione mal en un programa interfiera en el funcionamiento de otro.

Aunque la memoria asignada a procesos específicos normalmente está aislada, a veces los procesos necesitan poder compartir información. La memoria compartida es una de las técnicas más rápidas para la comunicación entre procesos .

La memoria se clasifica generalmente según la velocidad de acceso en almacenamiento primario y almacenamiento secundario . Los sistemas de gestión de memoria, entre otras operaciones, también se encargan del traslado de información entre estos dos niveles de memoria.

Gestión de memoria en OS/360 y sucesores

IBM System/360 no admite memoria virtual. [nota 5] El aislamiento de memoria de los trabajos se logra opcionalmente mediante claves de protección , asignando a cada trabajo una clave de almacenamiento diferente, 0 para el supervisor o 1–15. La gestión de memoria en OS/360 es una función del supervisorGETMAIN . El almacenamiento se solicita mediante la macro y se libera mediante la FREEMAINmacro, lo que da como resultado una llamada al supervisor ( SVC ) para realizar la operación.

En OS/360 los detalles varían dependiendo de cómo se genera el sistema , por ejemplo, para PCP , MFT , MVT .

En OS/360 MVT, la subasignación dentro de la región de un trabajo o el Área de cola del sistema (SQA) compartida se basa en subgrupos , áreas de un tamaño múltiplo de 2 KB (el tamaño de un área protegida por una clave de protección). Los subgrupos se numeran del 0 al 255. [10] Dentro de una región, a los subgrupos se les asigna la protección de almacenamiento del trabajo o la clave del supervisor, la clave 0. Los subgrupos 0 a 127 reciben la clave del trabajo. Inicialmente, solo se crea el subgrupo cero y todas las solicitudes de almacenamiento del usuario se satisfacen desde el subgrupo 0, a menos que se especifique otro en la solicitud de memoria. Los subgrupos 250 a 255 se crean mediante solicitudes de memoria del supervisor en nombre del trabajo. A la mayoría de estos se les asigna la clave 0, aunque algunos obtienen la clave del trabajo. Los números de subgrupo también son relevantes en MFT, aunque los detalles son mucho más simples. [11] MFT utiliza particiones fijas redefinibles por el operador en lugar de regiones dinámicas y PCP tiene solo una partición.

Cada subgrupo se asigna mediante una lista de bloques de control que identifican los bloques de memoria asignados y libres dentro del subgrupo. La memoria se asigna encontrando un área libre de tamaño suficiente o asignando bloques adicionales en el subgrupo, hasta el tamaño de la región del trabajo. Es posible liberar toda o parte de un área de memoria asignada. [12]

Los detalles para OS/VS1 son similares [13] a los de MFT y MVT; los detalles para OS/VS2 son similares a los de MVT, excepto que el tamaño de página es de 4 KiB. Tanto para OS/VS1 como para OS/VS2, el área de cola del sistema (SQA) compartida no es paginable.

En MVS, el espacio de direcciones [14] incluye un área compartida paginable adicional, el Área de almacenamiento común (CSA), y dos áreas privadas adicionales, el área de cola del sistema local no paginable (LSQA) y el área de trabajo del sistema paginable (SWA). Además, las claves de almacenamiento 0 a 7 están reservadas para su uso por parte de código privilegiado.

Véase también

Notas

  1. ^ Sin embargo, el entorno de ejecución de un procesador de lenguaje puede subdividir la memoria adquirida dinámicamente del sistema operativo, por ejemplo, para implementar una pila.
  2. ^ En algunos sistemas operativos, por ejemplo, OS/360 , el almacenamiento libre se puede subdividir de varias maneras, por ejemplo, subgrupos en OS/360 , debajo de la línea, encima de la línea y encima de la barra en z/OS .
  3. ^ No debe confundirse con la estructura de datos del montón no relacionada .
  4. ^ Una implementación simplista de estas dos funciones se puede encontrar en el artículo "Gestión interna de la memoria". [3]
  5. ^ Excepto en el modelo 67

Referencias

  1. ^ ab Detlefs, D.; Dosser, A.; Zorn, B. (junio de 1994). "Costos de asignación de memoria en programas grandes de C y C++" (PDF) . Software: Práctica y experiencia . 24 (6): 527–542. CiteSeerX  10.1.1.30.3073 . doi :10.1002/spe.4380240602. S2CID  14214110.
  2. ^ "Asignación de almacenamiento principal" (PDF) . IBM Operating System/360 Concepts and Facilities (PDF) . IBM Systems Reference Library (Primera edición). IBM Corporation. 1965. p. 74 . Consultado el 3 de abril de 2019 .
  3. ^ Jonathan Bartlett. "Gestión de memoria interna". IBM DeveloperWorks .
  4. ^ Silberschatz, Abraham ; Galvin, Peter B. (2004). Conceptos de sistemas operativos . Wiley. ISBN 0-471-69466-5.
  5. ^ alloca(3)  –  Manual del programador de Linux – Funciones de la biblioteca
  6. ^ "_malloca". Documentación de Microsoft CRT .
  7. ^ "gnulib/malloca.h". GitHub . Consultado el 24 de noviembre de 2019 .
  8. ^ "glibc/include/alloca.h". Los espejos de Beren Minor. 23 de noviembre de 2019.
  9. ^ Tanenbaum, Andrew S. (1992). Sistemas operativos modernos . Englewood Cliffs, Nueva Jersey: Prentice-Hall. pág. 90. ISBN. 0-13-588187-0.
  10. ^ OS360Sup, págs. 82-85.
  11. ^ OS360Sup, págs. 82.
  12. ^ IBM Corporation (mayo de 1973). Program Logic: IBM System/360 Operating System MVT Supervisor (PDF) . págs. 107–137 . Consultado el 3 de abril de 2019 .
  13. ^ OSVS1Dig, pág. 2.37-2.39.
  14. ^ "Virtual Storage Layout" (PDF) . Introducción a OS/VS2 Release 2 (PDF) . Sistemas (primera edición). IBM . Marzo de 1973. pág. 37. GC28-0661-1 . Consultado el 15 de julio de 2024 .

Bibliografía

Soporte OS360
Servicios de supervisión del sistema operativo IBM System/360 y macroinstrucciones (PDF) . Biblioteca de referencia de sistemas IBM (octava edición). IBM . Septiembre de 1974. GC28-6646-7.
OSVS1Dig
OS/VS1 Programmer's Reference Digest Release 6 (PDF) . Sistemas (sexta edición). IBM . 15 de septiembre de 1976. GC24-5091-5 con TNL.

Enlaces externos

  • Referencia de gestión de memoria, guía para principiantes Asignación