Skip to content

Latest commit

 

History

History
36 lines (23 loc) · 2.43 KB

README.md

File metadata and controls

36 lines (23 loc) · 2.43 KB

ext-gcignore

This extension provides functions that make the PHP GC ignore a particular object's dependencies.

Why?

In recent PocketMine-MP development, I discovered that, as a heavily used object, Server frequently landed in the GC root buffer, despite being a long-lived object that can't be garbage-collected anyway. This is because objects in PHP get marked for GC scanning when their refcount decreases to some value above 0.

This happens in basically every interaction with an object, meaning that frequently-used objects will basically always get scanned for GC cycles, even though they can't be deleted anyway.

This gets even worse when complex objects are indirectly referenced by other frequently-used objects, as the costs merge together.

For example, PocketMine-MP World is, by itself, somewhat less costly to examine than Server. However, since World contains a reference to Server, all of Server's dependencies will get recursively scanned whenever a World is unref'd, which also happens frequently.

How to use

I recommend using gc_ignore() on any object that meets any of the following criteria:

  • References a lot of objects, directly or indirectly
  • Is a singleton

It may also be beneficial for simple, short-lived objects produced en-masse that you're sure are acyclic, like a Vector3. If you cache these types of objects, they'll fill up the GC root buffer despite not having any cycles (although this may be addressed by changes like php/php-src#17130). However, I haven't done much testing of this use-case.

How does it work?

gc_ignore() overrides zend_object->handlers for the provided object with a custom get_gc handler that returns dummy data for the GC to look at.

Out of an abundance of caution, the extension currently only permits this modification for objects which have standard Zend object handlers. This means that many extensions' classes can't be altered, but extension classes are usually simple anyway, and so aren't expected to benefit from this treatment anyway.

API

  • gc_ignore(object $object) : bool - marks an object to prevent GC from looking at its properties; returns false and emits a warning if the object can't be ignored
  • gc_unignore(object $object) : void - reverses changes made by gc_ignore() on an object; does nothing if the object was not ignored
  • gc_is_ignored(object $object) : bool - returns whether gc_ignore() made changes to the object