Heim  >  Artikel  >  Backend-Entwicklung  >  Eine kurze Diskussion des PHP-Quellcodes 32: Emalloc/Efree-Schicht und Heap-Schicht im PHP-Speicherpool

Eine kurze Diskussion des PHP-Quellcodes 32: Emalloc/Efree-Schicht und Heap-Schicht im PHP-Speicherpool

不言
不言Original
2018-06-29 09:54:521869Durchsuche

Dieser Artikel stellt hauptsächlich den PHP-Quellcode 32 vor: Emalloc/efree-Schicht und Heap-Schicht im PHP-Speicherpool. Jetzt kann ich ihn mit allen teilen, die ihn benötigen 🎜>Kurzer Vortrag über PHP-Quellcode 32: Emalloc/Efree-Schicht und Heap-Schicht im PHP-Speicherpool

Emalloc/EFree-Schicht ist die Struktur der obersten Ebene im gesamten Speichersystem. Sie wird über den Heap-Schicht-Austausch geleitet und verwendet PHPs eigene Speicherverwaltungsmechanismus. Wenn USE_ZEND_ALLOC auf 0 gesetzt ist, können malloc/free und andere Funktionen verwendet werden, um den Speicher direkt zu betreiben.

Hier analysieren wir die Interaktion zwischen der Emalloc/Efree-Schicht und der Heap-Schicht sowie den Speicherverwaltungsmechanismus der Heap-Schicht aus der Implementierung der Emalloc- und EFree-Funktionen.

【emalloc】

Die Emalloc-Funktion beginnt in Zeile 70 von zend_alloc.h.

emalloc ist ein Makro, das der Funktion _emalloc entspricht.
Wenn in der Funktion _emalloc der Speicherverwaltungsmechanismus von Zend nicht verwendet wird, wird die Funktion malloc direkt aufgerufen, andernfalls

_zend_mm_alloc_int
[emalloc() -> _emalloc() -> _zend_mm_alloc_int() ]

In der Funktion _zend_mm_alloc_int verarbeitet das Programm, ob der tatsächlich erforderliche Speicher kleiner als oder ist größer oder gleich Wenn es in den beiden Fällen von ZEND_MM_MAX_SMALL_SIZE kleiner als ZEND_MM_MAX_SMALL_SIZE ist, wird in free_buckets nach einem geeigneten Speicherblock gesucht. Wenn in free_buckets ein geeigneter Block gefunden werden kann, springen Sie direkt zu zend_mm_finished_searching_for_block , andernfalls zend_mm_search_large_block()


[emalloc() -> _zend_mm_search_large_block()]

Die Funktion zend_mm_search_large_block wird verwendet in großen_freien_Buckets. Wenn die Größe ZEND_MM_LARGE_BUCKET_INDEX (true_size) nicht gefunden wird, müssen Sie den kleinsten Block in der größeren Blockliste finden.


Wenn es weder in der großen Blockliste noch in der kleinen Blockliste gefunden wird, müssen Sie in den verbleibenden Listenblöcken suchen. Wenn es gefunden wird, wird auch zu zend_mm_finished_searching_for_block gesprungen.

Wenn ja nicht in den drei Listen gefunden wird, muss die Speicherzuweisung erneut erhöht werden. Zu diesem Zeitpunkt wird die Zuweisungsfunktion der Speicherschicht aufgerufen. Wenn der zuzuweisende Speicher größer als block_size ist, muss er basierend auf der Größe neu berechnet werden direkt zugeordnet.

Nach dem Zuweisen von Speicher muss der Heap neu angeordnet werden. Zu diesem Zeitpunkt muss die Speichergröße im Heap neu berechnet werden und der neu zugewiesene Speicher wird am Anfang der segment_list hinzugefügt.

Wenn Sie im obigen Vorgang direkt zu zend_mm_finished_searching_for_block springen, müssen Sie den verwendeten Speicherblock aus der entsprechenden Liste entfernen (dies sollte ein Markierungsprozess, eine Pseudoentfernung) sein.

Als nächstes, abhängig von Wie viel Speicher noch übrig ist, verschieben Sie ihn in die freie Liste oder die verbleibende Liste.

Gibt schließlich den zugewiesenen Block zurück.

Während des gesamten Prozesses von emalloc sind im Folgenden einige Punkte zu beachten.

ZEND_MM_BUCKET_INDEX (true_size) befindet sich an der Position im Bucket. Dieser Wert ist größer oder gleich 0 und kleiner als 32.

Die Implementierung ist wie folgt:

#define ZEND_MM_BUCKET_INDEX(true_size) ((true_size>>ZEND_MM_ALIGNMENT_LOG2)-(ZEND_MM_ALIGNED_MIN_HEADER_SIZE>>ZEND_MM_ALIGNMENT_LOG2))

Die Werte von free_bitmap und large_free_bitmap sind beide 0 bis 31.


[efree]

Die efree-Funktion beginnt in Zeile 72 von zend_alloc.h.

efree ist ein Makro, das der Funktion _efree entspricht.
Wenn in der _efree-Funktion der Speicherverwaltungsmechanismus von Zend nicht verwendet wird, rufen Sie die kostenlose Funktion direkt auf, andernfalls rufen Sie _ auf.

zend_mm_free_int
[efree() -> _efree() -> _zend_mm_free_int() ]

Der Heap reduziert zuerst die Größe des gesamten Heaps, wenn der nächste Block des Der aktuelle Block ist ein freier Block. Löschen Sie dann den nächsten freien Block aus der Liste der freien Blöcke und führen Sie ihn mit dem aktuellen Block zusammen. Wenn der vorherige Block des aktuellen Blocks ein freier Block ist, löschen Sie den vorherigen freien Block aus der Liste der freien Blöcke Füge es mit dem aktuellen Block zusammen, der Zeiger zeigt auf den vorherigen freien Block. Wenn der aktuelle Block zu diesem Zeitpunkt der Startblock ist, rufen Sie zend_mm_del_segment auf, um das gesamte Speichersegment zu löschen. Wenn es sich nicht um den Startblock handelt, wird der zusammengeführte Block zur Liste der freien Blöcke hinzugefügt.

Das Obige ist der gesamte Inhalt dieses Artikels. Ich hoffe, dass er für das Studium aller hilfreich ist. Weitere verwandte Inhalte finden Sie auf der chinesischen PHP-Website.

Verwandte Empfehlungen:

Eine kurze Diskussion des PHP-Quellcodes 31: Grundlagen der Heap-Schicht im PHP-Speicherpool


Eine kurze Diskussion zu PHP-Quellcode 30: Die Speicherschicht im PHP-Speicherpool


Eine kurze Diskussion zu PHP-Quellcode 29: Über die Vererbung von Schnittstellen

Das obige ist der detaillierte Inhalt vonEine kurze Diskussion des PHP-Quellcodes 32: Emalloc/Efree-Schicht und Heap-Schicht im PHP-Speicherpool. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Stellungnahme:
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn