1
0
mirror of https://github.com/danog/parallel.git synced 2024-12-11 08:39:48 +01:00
parallel/src/Threading/Thread.php

226 lines
6.2 KiB
PHP
Raw Normal View History

2015-07-14 00:30:59 +02:00
<?php
namespace Icicle\Concurrent\Threading;
2015-08-27 16:10:08 +02:00
use Icicle\Concurrent\ContextInterface;
use Icicle\Concurrent\Exception\InvalidArgumentError;
use Icicle\Concurrent\Exception\StatusError;
use Icicle\Concurrent\Exception\SynchronizationError;
2015-08-28 23:18:02 +02:00
use Icicle\Concurrent\Exception\ThreadException;
use Icicle\Concurrent\Sync\Channel;
use Icicle\Concurrent\Sync\Internal\ExitStatusInterface;
use Icicle\Concurrent\SynchronizableInterface;
use Icicle\Coroutine;
2015-08-25 16:37:22 +02:00
use Icicle\Socket\Stream\DuplexStream;
2015-07-27 00:53:00 +02:00
/**
* Implements an execution context using native multi-threading.
2015-08-11 00:38:58 +02:00
*
* The thread context is not itself threaded. A local instance of the context is
* maintained both in the context that creates the thread and in the thread
* itself.
*/
class Thread implements ContextInterface, SynchronizableInterface
2015-07-14 00:30:59 +02:00
{
2015-08-25 16:39:58 +02:00
const LATENCY_TIMEOUT = 0.01; // 10 ms
2015-07-27 00:53:00 +02:00
/**
2015-08-31 00:52:00 +02:00
* @var Internal\Thread An internal thread instance.
2015-07-27 00:53:00 +02:00
*/
private $thread;
2015-07-27 00:53:00 +02:00
/**
2015-08-31 00:52:00 +02:00
* @var Channel A channel for communicating with the thread.
*/
private $channel;
/**
* @var resource
*/
private $socket;
/**
* @var bool
*/
private $started = false;
2015-08-07 01:59:25 +02:00
/**
* Spawns a new thread and runs it.
*
2015-08-31 00:52:00 +02:00
* @param callable $function The callable to invoke in the thread.
*
2015-08-31 00:52:00 +02:00
* @return Thread The thread object that was spawned.
2015-08-07 01:59:25 +02:00
*/
public static function spawn(callable $function /* , ...$args */)
{
2015-08-24 17:47:36 +02:00
$class = new \ReflectionClass(__CLASS__);
$thread = $class->newInstanceArgs(func_get_args());
$thread->start();
return $thread;
}
2015-08-18 17:12:06 +02:00
/**
2015-08-31 00:52:00 +02:00
* Creates a new thread.
*
2015-08-31 00:52:00 +02:00
* @param callable $function The callable to invoke in the thread when run.
*
* @throws InvalidArgumentError If the given function cannot be safely invoked in a thread.
2015-08-18 17:12:06 +02:00
*/
public function __construct(callable $function /* , ...$args */)
{
$args = array_slice(func_get_args(), 1);
// Make sure closures don't `use` other variables or have statics.
if ($function instanceof \Closure) {
$reflector = new \ReflectionFunction($function);
if (!empty($reflector->getStaticVariables())) {
throw new InvalidArgumentError('Closures with static variables cannot be passed to thread.');
}
}
list($channel, $this->socket) = Channel::createSocketPair();
$this->thread = new Internal\Thread($this->socket, $function, $args);
2015-08-25 16:37:22 +02:00
$this->channel = new Channel(new DuplexStream($channel));
}
2015-08-18 17:12:06 +02:00
2015-07-27 00:53:00 +02:00
/**
* Checks if the context is running.
2015-07-27 00:53:00 +02:00
*
* @return bool True if the context is running, otherwise false.
2015-07-27 00:53:00 +02:00
*/
public function isRunning()
{
return $this->thread->isRunning() && $this->channel->isOpen();
2015-08-07 01:59:25 +02:00
}
/**
2015-08-31 00:52:00 +02:00
* Spawns the thread and begins the thread's execution.
*
* @throws StatusError If the thread has already been started.
* @throws ThreadException If starting the thread was unsuccessful.
*/
public function start()
2015-07-14 00:30:59 +02:00
{
if ($this->started) {
throw new StatusError('The thread has already been started.');
}
if (!$this->thread->start(PTHREADS_INHERIT_INI | PTHREADS_INHERIT_FUNCTIONS | PTHREADS_INHERIT_CLASSES)) {
throw new ThreadException('Failed to start the thread.');
}
$this->started = true;
}
2015-08-11 00:38:58 +02:00
/**
* Immediately kills the context.
2015-08-31 00:52:00 +02:00
*
* @throws ThreadException If killing the thread was unsuccessful.
*/
public function kill()
{
if ($this->isRunning()) {
try {
if (!$this->thread->kill()) {
throw new ThreadException('Failed to kill the thread.');
}
} finally {
$this->channel->close();
fclose($this->socket);
}
2015-08-28 23:18:02 +02:00
}
}
2015-08-18 17:12:06 +02:00
/**
* @coroutine
*
* Gets a promise that resolves when the context ends and joins with the
* parent context.
2015-08-18 17:12:06 +02:00
*
* @return \Generator
*
* @resolve mixed Resolved with the return or resolution value of the context once it has completed execution.
*
2015-08-31 00:52:00 +02:00
* @throws StatusError Thrown if the context has not been started.
* @throws SynchronizationError Thrown if an exit status object is not received.
2015-08-18 17:12:06 +02:00
*/
public function join()
2015-08-18 17:12:06 +02:00
{
if (!$this->started) {
throw new StatusError('The thread has not been started.');
}
2015-08-18 17:12:06 +02:00
try {
$response = (yield $this->channel->receive());
if (!$response instanceof ExitStatusInterface) {
throw new SynchronizationError('Did not receive an exit status from thread.');
2015-08-18 17:12:06 +02:00
}
yield $response->getResult();
2015-08-18 17:12:06 +02:00
} finally {
$this->thread->join();
$this->channel->close();
fclose($this->socket);
2015-08-18 17:12:06 +02:00
}
}
/**
* {@inheritdoc}
2015-08-18 17:12:06 +02:00
*/
public function receive()
2015-08-18 17:12:06 +02:00
{
if (!$this->started) {
throw new StatusError('The thread has not been started.');
}
$data = (yield $this->channel->receive());
if ($data instanceof ExitStatusInterface) {
$data = $data->getResult();
throw new SynchronizationError(sprintf(
'Thread unexpectedly exited with result of type: %s',
is_object($data) ? get_class($data) : gettype($data)
));
}
yield $data;
2015-08-18 17:12:06 +02:00
}
2015-08-07 01:59:25 +02:00
/**
* {@inheritdoc}
2015-08-07 01:59:25 +02:00
*/
public function send($data)
{
if (!$this->started) {
throw new StatusError('The thread has not been started.');
}
if ($data instanceof ExitStatusInterface) {
throw new InvalidArgumentError('Cannot send exit status objects.');
2015-07-27 00:53:00 +02:00
}
2015-07-14 00:30:59 +02:00
yield $this->channel->send($data);
}
/**
* {@inheritdoc}
*/
public function synchronized(callable $callback)
{
if (!$this->started) {
throw new StatusError('The thread has not been started.');
}
while (!$this->thread->tsl()) {
2015-08-25 16:39:58 +02:00
yield Coroutine\sleep(self::LATENCY_TIMEOUT);
2015-08-19 02:12:58 +02:00
}
try {
yield $callback($this);
} finally {
$this->thread->release();
}
}
2015-07-14 00:30:59 +02:00
}