key = abs(crc32(spl_object_hash($this))); $this->memOpen($this->key, 'n', $permissions, $size + self::MEM_DATA_OFFSET); $this->setHeader(self::STATE_ALLOCATED, 0, $permissions); $this->set($value); } /** * Gets the shared value stored in the container. * * @return mixed The stored value. */ public function deref() { if ($this->isFreed()) { throw new SharedMemoryException('The object has already been freed.'); } $header = $this->getHeader(); // Make sure the header is in a valid state and format. if ($header['state'] !== self::STATE_ALLOCATED || $header['size'] <= 0) { throw new SharedMemoryException('Shared object memory is corrupt.'); } // Read the actual value data from memory and unserialize it. $data = $this->memGet(self::MEM_DATA_OFFSET, $header['size']); return unserialize($data); } /** * Sets the value in the container to a new value. * * @param mixed $value The value to set. */ public function set($value) { if ($this->isFreed()) { throw new SharedMemoryException('The object has already been freed.'); } $serialized = serialize($value); $size = strlen($serialized); $header = $this->getHeader(); // If we run out of space, we need to allocate a new shared memory // segment that is larger than the current one. To coordinate with other // processes, we will leave a message in the old segment that the segment // has moved and along with the new key. The old segment will be discarded // automatically after all other processes notice the change and close // the old handle. if (shmop_size($this->handle) < $size + self::MEM_DATA_OFFSET) { $this->key = $this->key < 0xffffffff ? $this->key + 1 : mt_rand(0x10, 0xfffffffe); $this->setHeader(self::STATE_MOVED, $this->key, 0); $this->memDelete(); shmop_close($this->handle); $this->memOpen($this->key, 'n', $header['permissions'], $size * 2); } // Rewrite the header and the serialized value to memory. $this->setHeader(self::STATE_ALLOCATED, $size, $header['permissions']); $this->memSet(self::MEM_DATA_OFFSET, $serialized); } /** * Frees the shared object from memory. * * The memory containing the shared value will be invalidated. When all * process disconnect from the object, the shared memory block will be * destroyed. * * Calling `free()` on an object already freed will have no effect. */ public function free() { if (!$this->isFreed()) { // Invalidate the memory block by setting its state to FREED. $this->setHeader(static::STATE_FREED, 0, 0); // Request the block to be deleted, then close our local handle. $this->memDelete(); shmop_close($this->handle); $this->handle = null; } } /** * Checks if the object has been freed. * * Note that this does not check if the object has been destroyed; it only * checks if this handle has freed its reference to the object. * * @return bool True if the object is freed, otherwise false. */ public function isFreed() { // If we are no longer connected to the memory segment, check if it has // been invalidated. if ($this->handle !== null) { $this->handleMovedMemory(); $header = $this->getHeader(); return $header['state'] === static::STATE_FREED; } return true; } /** * Serializes the local object handle. * * Note that this does not serialize the object that is referenced, just the * object handle. * * @return string The serialized object handle. */ public function serialize() { return serialize($this->key); } /** * Unserializes the local object handle. * * @param string $serialized The serialized object handle. */ public function unserialize($serialized) { $this->key = unserialize($serialized); $this->memOpen($this->key, 'w', 0, 0); } /** * Handles cloning, which creates clones the local object and creates a new * local object handle. */ public function __clone() { $value = $this->deref(); $this->__construct($value); } /** * Gets information about the object for debugging purposes. * * @return array An array of debugging information. */ public function __debugInfo() { if ($this->isFreed()) { return [ 'id' => $this->key, 'object' => null, 'freed' => true, ]; } return [ 'id' => $this->key, 'object' => $this->deref(), 'freed' => false, ]; } /** * Updates the current memory segment handle, handling any moves made on the * data. */ private function handleMovedMemory() { // Read from the memory block and handle moved blocks until we find the // correct block. while (true) { $header = $this->getHeader(); // If the state is STATE_MOVED, the memory is stale and has been moved // to a new location. Move handle and try to read again. if ($header['state'] !== self::STATE_MOVED) { break; } shmop_close($this->handle); $this->key = $header['size']; $this->memOpen($this->key, 'w', 0, 0); } } /** * Reads and returns the data header at the current memory segment. * * @return array An associative array of header data. */ private function getHeader() { $data = $this->memGet(0, self::MEM_DATA_OFFSET); return unpack('Cstate/Lsize/Spermissions', $data); } /** * Sets the header data for the current memory segment. * * @param int $state An object state. * @param int $size The size of the stored data, or other value. * @param int $permissions The permissions mask on the memory segment. */ private function setHeader($state, $size, $permissions) { $header = pack('CLS', $state, $size, $permissions); $this->memSet(0, $header); } /** * Opens a shared memory handle. * * @param int $key The shared memory key. * @param string $mode The mode to open the shared memory in. * @param int $permissions Process permissions on the shared memory. * @param int $size The size to crate the shared memory in bytes. */ private function memOpen($key, $mode, $permissions, $size) { $this->handle = @shmop_open($key, $mode, $permissions, $size); if ($this->handle === false) { throw new SharedMemoryException('Failed to create shared memory block.'); } } /** * Reads binary data from shared memory. * * @param int $offset The offset to read from. * @param int $size The number of bytes to read. * * @return string The binary data at the given offset. */ private function memGet($offset, $size) { $data = shmop_read($this->handle, $offset, $size); if ($data === false) { throw new SharedMemoryException('Failed to read from shared memory block.'); } return $data; } /** * Writes binary data to shared memory. * * @param int $offset The offset to write to. * @param string $data The binary data to write. */ private function memSet($offset, $data) { if (!shmop_write($this->handle, $data, $offset)) { throw new SharedMemoryException('Failed to write to shared memory block.'); } } /** * Requests the shared memory segment to be deleted. */ private function memDelete() { if (!shmop_delete($this->handle)) { throw new SharedMemoryException('Failed to discard shared memory block.'); } } }