mirror of
https://github.com/danog/parallel.git
synced 2025-01-22 22:11:11 +01:00
Remove old POSIX semaphores & shared objects
This commit is contained in:
parent
ec3f5621b7
commit
3c531d2d02
@ -1,280 +0,0 @@
|
||||
<?php
|
||||
namespace Icicle\Concurrent\Forking;
|
||||
|
||||
use Icicle\Concurrent\Exception\SynchronizedMemoryException;
|
||||
|
||||
/**
|
||||
* A synchronized object that safely shares its state across processes and
|
||||
* provides methods for process synchronization.
|
||||
*
|
||||
* When used with forking, the object must be created prior to forking for both
|
||||
* processes to access the synchronized object.
|
||||
*/
|
||||
abstract class SharedObject
|
||||
{
|
||||
/**
|
||||
* @var int The default amount of bytes to allocate for the object.
|
||||
*/
|
||||
const SHM_DEFAULT_SIZE = 16384;
|
||||
|
||||
/**
|
||||
* @var int The byte offset to the start of the object data in memory.
|
||||
*/
|
||||
const SHM_DATA_OFFSET = 5;
|
||||
|
||||
/**
|
||||
* @var int The default permissions for other processes to access the object.
|
||||
*/
|
||||
const OBJECT_PERMISSIONS = 0600;
|
||||
|
||||
/**
|
||||
* @var The shared memory segment key.
|
||||
*/
|
||||
private $__key;
|
||||
|
||||
/**
|
||||
* @var An open handle to the shared memory segment.
|
||||
*/
|
||||
private $__shm;
|
||||
|
||||
/**
|
||||
* @var array A local cache of property values that are synchronized.
|
||||
*/
|
||||
private $__synchronizedProperties = [];
|
||||
|
||||
/**
|
||||
* Creates a new synchronized object.
|
||||
*/
|
||||
public function __construct()
|
||||
{
|
||||
$this->__key = abs(crc32(spl_object_hash($this)));
|
||||
$this->__open($this->__key, 'c', static::OBJECT_PERMISSIONS, static::SHM_DEFAULT_SIZE);
|
||||
$this->__write(0, pack('x5'));
|
||||
$this->__initSynchronizedProperties();
|
||||
}
|
||||
|
||||
/**
|
||||
* Destroys the synchronized object.
|
||||
*/
|
||||
public function destroy()
|
||||
{
|
||||
if (!shmop_delete($this->__shm)) {
|
||||
throw new SynchronizedMemoryException('Failed to discard shared memory block.');
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Checks if a synchronized property is set.
|
||||
*
|
||||
* @param string $name The name of the property to check.
|
||||
*
|
||||
* @return bool True if the property is set, otherwise false.
|
||||
*
|
||||
* @internal
|
||||
*/
|
||||
final public function __isset($name)
|
||||
{
|
||||
$this->__readSynchronizedProperties();
|
||||
return isset($this->__synchronizedProperties[$name]);
|
||||
}
|
||||
|
||||
/**
|
||||
* Gets the value of a synchronized property.
|
||||
*
|
||||
* @param string $name The name of the property to get.
|
||||
*
|
||||
* @return mixed The value of the property.
|
||||
*
|
||||
* @internal
|
||||
*/
|
||||
final public function __get($name)
|
||||
{
|
||||
$this->__readSynchronizedProperties();
|
||||
return $this->__synchronizedProperties[$name];
|
||||
}
|
||||
|
||||
/**
|
||||
* Sets the value of a synchronized property.
|
||||
*
|
||||
* @param string $name The name of the property to set.
|
||||
* @param mixed $value The value to set the property to.
|
||||
*
|
||||
* @internal
|
||||
*/
|
||||
final public function __set($name, $value)
|
||||
{
|
||||
$this->__readSynchronizedProperties();
|
||||
$this->__synchronizedProperties[$name] = $value;
|
||||
$this->__writeSynchronizedProperties();
|
||||
}
|
||||
|
||||
/**
|
||||
* Unsets a synchronized property.
|
||||
*
|
||||
* @param string $name The name of the property to unset.
|
||||
*
|
||||
* @internal
|
||||
*/
|
||||
final public function __unset($name)
|
||||
{
|
||||
$this->__readSynchronizedProperties();
|
||||
if (isset($this->__synchronizedProperties[$name])) {
|
||||
unset($this->__synchronizedProperties[$name]);
|
||||
$this->__writeSynchronizedProperties();
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Initializes the internal synchronized property table.
|
||||
*
|
||||
* This method does some ugly hackery to put on a nice face elsewhere. At
|
||||
* call-time, the descendant type's defined and inherited properties are
|
||||
* scanned for \@synchronized annotations.
|
||||
*
|
||||
* @internal
|
||||
*/
|
||||
private function __initSynchronizedProperties()
|
||||
{
|
||||
$class = new \ReflectionClass(get_called_class());
|
||||
$synchronizedProperties = [];
|
||||
|
||||
// Find *all* defined and inherited properties of the called class (late
|
||||
// binding) and get which class the property was defined in. This
|
||||
// includes inherited private properties.
|
||||
do {
|
||||
foreach ($class->getProperties() as $property) {
|
||||
if (!$property->isStatic()) {
|
||||
$comment = $property->getDocComment();
|
||||
if ($comment && strpos($comment, '@synchronized') !== false) {
|
||||
$synchronizedProperties[$property->getName()] = $class->getName();
|
||||
}
|
||||
}
|
||||
}
|
||||
} while ($class = $class->getParentClass());
|
||||
|
||||
// Define a closure that deletes a property and returns its default
|
||||
// value. This function will be called on the current object to delete
|
||||
// synchronized properties (by being bound to the class scope that
|
||||
// defined the property).
|
||||
$unsetter = function ($name) {
|
||||
$initValue = $this->{$name};
|
||||
unset($this->{$name});
|
||||
return $initValue;
|
||||
};
|
||||
|
||||
// Cache the synchronized property table.
|
||||
foreach ($synchronizedProperties as $property => $class) {
|
||||
$this->__synchronizedProperties[$property] = $unsetter
|
||||
->bindTo($this, $class)
|
||||
->__invoke($property);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Reloads the object's property table from shared memory.
|
||||
*
|
||||
* @internal
|
||||
*/
|
||||
private function __readSynchronizedProperties()
|
||||
{
|
||||
$data = $this->__read(0, static::SHM_DATA_OFFSET);
|
||||
$header = unpack('Cstate/Lsize', $data);
|
||||
|
||||
// State set to 1 indicates the memory is stale and has been moved to a
|
||||
// new location. Move handle and try to read again.
|
||||
if ($header['state'] === 1) {
|
||||
shmop_close($this->__shm);
|
||||
$this->__key = $header['size'];
|
||||
$this->__open($this->__key, 'w', 0, 0);
|
||||
$this->__readSynchronizedProperties();
|
||||
return;
|
||||
}
|
||||
|
||||
if ($header['size'] > 0) {
|
||||
$data = $this->__read(static::SHM_DATA_OFFSET, $header['size']);
|
||||
$this->__synchronizedProperties = unserialize($data);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Writes the object's property table to shared memory.
|
||||
*
|
||||
* @internal
|
||||
*/
|
||||
protected function __writeSynchronizedProperties()
|
||||
{
|
||||
$serialized = serialize($this->__synchronizedProperties);
|
||||
$size = strlen($serialized);
|
||||
|
||||
// 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->__shm) < $size + static::SHM_DATA_OFFSET) {
|
||||
$this->__key = $this->__key < 0xffffffff ? $this->__key + 1 : rand(0x10, 0xfffffffe);
|
||||
$header = pack('CL', 1, $this->__key);
|
||||
$this->__write(0, $header);
|
||||
$this->destroy();
|
||||
shmop_close($this->__shm);
|
||||
|
||||
$this->__open($this->__key, 'c', static::OBJECT_PERMISSIONS, $size * 2);
|
||||
}
|
||||
|
||||
$data = pack('xLa*', $size, $serialized);
|
||||
$this->__write(0, $data);
|
||||
}
|
||||
|
||||
/**
|
||||
* 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.
|
||||
*
|
||||
* @internal
|
||||
*/
|
||||
private function __open($key, $mode, $permissions, $size)
|
||||
{
|
||||
$this->__shm = shmop_open($key, $mode, $permissions, $size);
|
||||
if ($this->__shm === false) {
|
||||
throw new SynchronizedMemoryException('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.
|
||||
*
|
||||
* @internal
|
||||
*/
|
||||
private function __read($offset, $size)
|
||||
{
|
||||
$data = shmop_read($this->__shm, $offset, $size);
|
||||
if ($data === false) {
|
||||
throw new SynchronizedMemoryException('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.
|
||||
*
|
||||
* @internal
|
||||
*/
|
||||
private function __write($offset, $data)
|
||||
{
|
||||
if (!shmop_write($this->__shm, $data, $offset)) {
|
||||
throw new SynchronizedMemoryException('Failed to write to shared memory block.');
|
||||
}
|
||||
}
|
||||
}
|
@ -1,76 +0,0 @@
|
||||
<?php
|
||||
namespace Icicle\Concurrent\Forking;
|
||||
|
||||
use Icicle\Concurrent\SynchronizableInterface;
|
||||
use Icicle\Concurrent\Sync\AsyncSemaphore;
|
||||
|
||||
/**
|
||||
* A synchronized object that safely shares its state across processes and
|
||||
* provides methods for process synchronization.
|
||||
*
|
||||
* When used with forking, the object must be created prior to forking for both
|
||||
* processes to access the synchronized object.
|
||||
*/
|
||||
class Synchronized extends SharedObject implements SynchronizableInterface
|
||||
{
|
||||
/**
|
||||
* @var AsyncSemaphore A semaphore used for locking the object data.
|
||||
*/
|
||||
private $semaphore;
|
||||
|
||||
/**
|
||||
* Creates a new synchronized object.
|
||||
*/
|
||||
public function __construct()
|
||||
{
|
||||
parent::__construct();
|
||||
$this->semaphore = new AsyncSemaphore(1);
|
||||
}
|
||||
|
||||
/**
|
||||
* Locks the object for read or write for the calling context.
|
||||
*/
|
||||
public function lock()
|
||||
{
|
||||
return $this->semaphore->acquire();
|
||||
}
|
||||
|
||||
/**
|
||||
* Unlocks the object.
|
||||
*/
|
||||
public function unlock()
|
||||
{
|
||||
$this->__writeSynchronizedProperties();
|
||||
return $this->semaphore->release();
|
||||
}
|
||||
|
||||
/**
|
||||
* Invokes a function while maintaining a lock for the calling context.
|
||||
*
|
||||
* @param callable $callback The function to invoke.
|
||||
*
|
||||
* @return mixed The value returned by the callback.
|
||||
*/
|
||||
public function synchronized(callable $callback)
|
||||
{
|
||||
return $this->lock()->then(function () use ($callback) {
|
||||
try {
|
||||
$returnValue = $callback($this);
|
||||
} finally {
|
||||
$this->unlock();
|
||||
}
|
||||
|
||||
return $returnValue;
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* Destroys the synchronized object safely on destruction.
|
||||
*/
|
||||
public function __destruct()
|
||||
{
|
||||
/*$this->synchronized(function () {
|
||||
$this->destroy();
|
||||
});*/
|
||||
}
|
||||
}
|
@ -1,168 +0,0 @@
|
||||
<?php
|
||||
namespace Icicle\Concurrent\Sync;
|
||||
|
||||
use Icicle\Concurrent\Exception\InvalidArgumentError;
|
||||
use Icicle\Concurrent\Exception\SemaphoreException;
|
||||
use Icicle\Concurrent\Forking\SharedObject;
|
||||
use Icicle\Loop;
|
||||
use Icicle\Promise;
|
||||
|
||||
/**
|
||||
* An asynchronous semaphore with non-blocking lock requests.
|
||||
*
|
||||
* To keep in sync with all handles to the async semaphore, a synchronous
|
||||
* semaphore is used as a gatekeeper to access the lock count; such locks are
|
||||
* guaranteed to perform very few memory read or write operations to reduce the
|
||||
* semaphore latency.
|
||||
*/
|
||||
class AsyncSemaphore extends SharedObject
|
||||
{
|
||||
/**
|
||||
* @var int The number of available locks.
|
||||
* @synchronized
|
||||
*/
|
||||
private $locks;
|
||||
|
||||
/**
|
||||
* @var int The maximum number of locks the semaphore allows.
|
||||
* @synchronized
|
||||
*/
|
||||
private $maxLocks;
|
||||
|
||||
/**
|
||||
* @var int A queue of processes waiting on locks.
|
||||
* @synchronized
|
||||
*/
|
||||
private $processQueue;
|
||||
|
||||
/**
|
||||
* @var \SplQueue A queue of promises waiting to acquire a lock within the
|
||||
* current calling context.
|
||||
*/
|
||||
private $waitQueue;
|
||||
|
||||
/**
|
||||
* @var Semaphore A synchronous semaphore for double locking.
|
||||
*/
|
||||
private $semaphore;
|
||||
|
||||
/**
|
||||
* Creates a new asynchronous semaphore.
|
||||
*
|
||||
* @param int $maxLocks The maximum number of processes that can lock the semaphore.
|
||||
*/
|
||||
public function __construct($maxLocks = 1)
|
||||
{
|
||||
parent::__construct();
|
||||
|
||||
if (!is_int($maxLocks) || $maxLocks < 1) {
|
||||
throw new InvalidArgumentError('Max locks must be a positive integer.');
|
||||
}
|
||||
|
||||
$this->locks = $maxLocks;
|
||||
$this->maxLocks = $maxLocks;
|
||||
$this->processQueue = new \SplQueue();
|
||||
$this->waitQueue = new \SplQueue();
|
||||
$this->semaphore = new Semaphore(1);
|
||||
|
||||
Loop\signal(SIGUSR1, function () {
|
||||
$this->handlePendingLocks();
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* Acquires a lock from the semaphore.
|
||||
*
|
||||
* @return PromiseInterface A promise resolved when a lock has been acquired.
|
||||
*
|
||||
* If there are one or more locks available, the returned promise is resolved
|
||||
* immediately and the lock count is decreased. If no locks are available,
|
||||
* the semaphore waits asynchronously for an unlock signal from another
|
||||
* process before resolving.
|
||||
*/
|
||||
public function acquire()
|
||||
{
|
||||
$deferred = new Promise\Deferred();
|
||||
|
||||
// Alright, we gotta get in and out as fast as possible. Deep breath...
|
||||
$this->semaphore->acquire();
|
||||
|
||||
try {
|
||||
if ($this->locks > 0) {
|
||||
// Oh goody, a free lock! Acquire a lock and get outta here!
|
||||
--$this->locks;
|
||||
$deferred->resolve();
|
||||
} else {
|
||||
$this->waitQueue->enqueue($deferred);
|
||||
$this->processQueue->enqueue(getmypid());
|
||||
$this->__writeSynchronizedProperties();
|
||||
}
|
||||
} finally {
|
||||
$this->semaphore->release();
|
||||
}
|
||||
|
||||
return $deferred->getPromise();
|
||||
}
|
||||
|
||||
/**
|
||||
* Releases a lock to the semaphore.
|
||||
*
|
||||
* @return PromiseInterface A promise resolved when a lock has been released.
|
||||
*
|
||||
* Note that this function is near-atomic and returns almost immediately. A
|
||||
* promise is returned only for consistency.
|
||||
*/
|
||||
public function release()
|
||||
{
|
||||
$this->semaphore->acquire();
|
||||
|
||||
if ($this->locks === $this->maxLocks) {
|
||||
$this->semaphore->release();
|
||||
throw new SemaphoreException('No locks acquired to release.');
|
||||
}
|
||||
|
||||
++$this->locks;
|
||||
|
||||
if (!$this->processQueue->isEmpty()) {
|
||||
$pid = $this->processQueue->dequeue();
|
||||
$pending = true;
|
||||
}
|
||||
|
||||
$this->semaphore->release();
|
||||
|
||||
if ($pending) {
|
||||
if ($pid === getmypid()) {
|
||||
$this->waitQueue->dequeue()->resolve();
|
||||
} else {
|
||||
posix_kill($pid, SIGUSR1);
|
||||
}
|
||||
}
|
||||
|
||||
return Promise\resolve();
|
||||
}
|
||||
|
||||
/**
|
||||
* Handles pending lock requests and resolves a pending acquire() call if
|
||||
* new locks are available.
|
||||
*/
|
||||
private function handlePendingLocks()
|
||||
{
|
||||
$dequeue = false;
|
||||
$this->semaphore->acquire();
|
||||
if ($this->locks > 0 && !$this->waitQueue->isEmpty()) {
|
||||
--$this->locks;
|
||||
$dequeue = true;
|
||||
}
|
||||
$this->semaphore->release();
|
||||
|
||||
if ($dequeue) {
|
||||
$this->waitQueue->dequeue()->resolve();
|
||||
}
|
||||
}
|
||||
|
||||
public function destroy()
|
||||
{
|
||||
parent::destroy();
|
||||
$this->semaphore->destroy();
|
||||
}
|
||||
}
|
@ -17,6 +17,9 @@ use Icicle\Concurrent\Exception\SharedMemoryException;
|
||||
*
|
||||
* Note that accessing a shared object is not atomic. Access to a shared object
|
||||
* should be protected with a mutex to preserve data integrity.
|
||||
*
|
||||
* When used with forking, the object must be created prior to forking for both
|
||||
* processes to access the synchronized object.
|
||||
*/
|
||||
class Parcel implements ParcelInterface, \Serializable
|
||||
{
|
||||
|
@ -1,111 +0,0 @@
|
||||
<?php
|
||||
namespace Icicle\Concurrent\Sync;
|
||||
|
||||
use Icicle\Concurrent\Exception\SemaphoreException;
|
||||
|
||||
/**
|
||||
* A synchronous semaphore that uses System V IPC semaphores.
|
||||
*/
|
||||
class Semaphore implements SemaphoreInterface, \Serializable
|
||||
{
|
||||
/**
|
||||
* @var int The key to the semaphore.
|
||||
*/
|
||||
private $key;
|
||||
|
||||
/**
|
||||
* @var int The maximum number of locks.
|
||||
*/
|
||||
private $maxLocks;
|
||||
|
||||
/**
|
||||
* @var resource An open handle to the semaphore.
|
||||
*/
|
||||
private $handle;
|
||||
|
||||
/**
|
||||
* Creates a new semaphore with a given number of locks.
|
||||
*
|
||||
* @param int $maxLocks The maximum number of locks that can be acquired from the semaphore.
|
||||
* @param int $permissions Permissions to access the semaphore.
|
||||
*/
|
||||
public function __construct($maxLocks = 1, $permissions = 0600)
|
||||
{
|
||||
$this->key = abs(crc32(spl_object_hash($this)));
|
||||
$this->maxLocks = $maxLocks;
|
||||
$this->handle = sem_get($this->key, $maxLocks, $permissions, 1);
|
||||
|
||||
if ($this->handle === false) {
|
||||
throw new SemaphoreException('Failed to create the semaphore.');
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Gets the maximum number of locks.
|
||||
*
|
||||
* @return int The maximum number of locks.
|
||||
*/
|
||||
public function getMaxLocks()
|
||||
{
|
||||
return $this->maxLocks;
|
||||
}
|
||||
|
||||
/**
|
||||
* Acquires a lock from the semaphore.
|
||||
*
|
||||
* Blocks until a lock can be acquired.
|
||||
*/
|
||||
public function acquire()
|
||||
{
|
||||
if (!sem_acquire($this->handle)) {
|
||||
throw new SemaphoreException('Failed to lock the semaphore.');
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Releases a lock to the semaphore.
|
||||
*/
|
||||
public function release()
|
||||
{
|
||||
if (!sem_release($this->handle)) {
|
||||
throw new SemaphoreException('Failed to unlock the semaphore.');
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Removes the semaphore if it still exists.
|
||||
*/
|
||||
public function destroy()
|
||||
{
|
||||
if (!@sem_remove($this->handle)) {
|
||||
$error = error_get_last();
|
||||
|
||||
if ($error['type'] !== E_WARNING) {
|
||||
throw new SemaphoreException('Failed to remove the semaphore.');
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Serializes the semaphore.
|
||||
*
|
||||
* @return string The serialized semaphore.
|
||||
*/
|
||||
public function serialize()
|
||||
{
|
||||
return serialize([$this->key, $this->maxLocks]);
|
||||
}
|
||||
|
||||
/**
|
||||
* Unserializes a serialized semaphore.
|
||||
*
|
||||
* @param string $serialized The serialized semaphore.
|
||||
*/
|
||||
public function unserialize($serialized)
|
||||
{
|
||||
// Get the semaphore key and attempt to re-connect to the semaphore in
|
||||
// memory.
|
||||
list($this->key, $this->maxLocks) = unserialize($serialized);
|
||||
$this->handle = sem_get($this->key, $maxLocks, 0600, 1);
|
||||
}
|
||||
}
|
@ -2,19 +2,29 @@
|
||||
namespace Icicle\Concurrent\Sync;
|
||||
|
||||
/**
|
||||
* A counting semaphore interface.
|
||||
* A non-blocking counting semaphore.
|
||||
*
|
||||
* Objects that implement this interface should guarantee that all operations
|
||||
* are atomic.
|
||||
*/
|
||||
interface SemaphoreInterface
|
||||
interface SemaphoreInterface extends \Countable
|
||||
{
|
||||
/**
|
||||
* Gets the number of currently available locks.
|
||||
*
|
||||
* @return int The number of available locks.
|
||||
*/
|
||||
public function count();
|
||||
|
||||
/**
|
||||
* @coroutine
|
||||
*
|
||||
* Acquires a lock from the semaphore asynchronously.
|
||||
*
|
||||
* @return \Generator
|
||||
* If there are one or more locks available, this function resolve imsmediately with a lock and the lock count is
|
||||
* decreased. If no locks are available, the semaphore waits asynchronously for a lock to become available.
|
||||
*
|
||||
* @return \Generator Resolves with a lock object when the acquire is successful.
|
||||
*
|
||||
* @resolve \Icicle\Concurrent\Sync\Lock
|
||||
*/
|
||||
|
@ -2,7 +2,7 @@
|
||||
namespace Icicle\Concurrent;
|
||||
|
||||
/**
|
||||
* Interface for objects that can be synchronized across contexts.
|
||||
* An object that can be synchronized for exclusive access across contexts.
|
||||
*/
|
||||
interface SynchronizableInterface
|
||||
{
|
||||
@ -11,11 +11,13 @@ interface SynchronizableInterface
|
||||
*
|
||||
* Invokes a function while maintaining a lock on the object.
|
||||
*
|
||||
* @param callable $callback The function to invoke.
|
||||
* The given callback will be passed the object being synchronized on as the first argument.
|
||||
*
|
||||
* @param callable<self> $callback The synchronized function to invoke.
|
||||
*
|
||||
* @return \Generator
|
||||
*
|
||||
* @resolve mixed Return value of $callback.
|
||||
* @resolve mixed The return value of $callback.
|
||||
*/
|
||||
public function synchronized(callable $callback);
|
||||
}
|
||||
|
Loading…
x
Reference in New Issue
Block a user