mirror of
https://github.com/danog/amp.git
synced 2024-11-27 12:35:02 +01:00
670 lines
19 KiB
PHP
670 lines
19 KiB
PHP
<?php
|
|
|
|
namespace Amp\Loop;
|
|
|
|
use Amp\Loop\Internal\Watcher;
|
|
use Interop\Async\Loop\Driver;
|
|
use Interop\Async\Loop\Registry;
|
|
use Interop\Async\Loop\UnsupportedFeatureException;
|
|
|
|
class NativeLoop implements Driver {
|
|
use Registry;
|
|
|
|
const MILLISEC_PER_SEC = 1e3;
|
|
const MICROSEC_PER_SEC = 1e6;
|
|
|
|
/**
|
|
* @var string
|
|
*/
|
|
private $nextId = "a";
|
|
|
|
/**
|
|
* @var \Amp\Loop\Internal\Watcher[]
|
|
*/
|
|
private $watchers = [];
|
|
|
|
/**
|
|
* @var string[]
|
|
*/
|
|
private $deferQueue = [];
|
|
|
|
/**
|
|
* @var resource[]
|
|
*/
|
|
private $readStreams = [];
|
|
|
|
/**
|
|
* @var \Amp\Loop\Internal\Watcher[][]
|
|
*/
|
|
private $readWatchers = [];
|
|
|
|
/**
|
|
* @var resource[]
|
|
*/
|
|
private $writeStreams = [];
|
|
|
|
/**
|
|
* @var \Amp\Loop\Internal\Watcher[][]
|
|
*/
|
|
private $writeWatchers = [];
|
|
|
|
/**
|
|
* @var int[]
|
|
*/
|
|
private $timerExpires = [];
|
|
|
|
/**
|
|
* @var \SplPriorityQueue
|
|
*/
|
|
private $timerQueue;
|
|
|
|
/**
|
|
* @var \Amp\Loop\Internal\Watcher[][]
|
|
*/
|
|
private $signalWatchers = [];
|
|
|
|
/**
|
|
* @var \Amp\Loop\Internal\Watcher[]
|
|
*/
|
|
private $unreferenced = [];
|
|
|
|
/**
|
|
* @var callable
|
|
*/
|
|
private $errorHandler;
|
|
|
|
/**
|
|
* @var bool
|
|
*/
|
|
private $running = false;
|
|
|
|
/**
|
|
* @var bool
|
|
*/
|
|
private $signalHandling;
|
|
|
|
public function __construct() {
|
|
$this->timerQueue = new \SplPriorityQueue();
|
|
$this->signalHandling = \extension_loaded("pcntl");
|
|
}
|
|
|
|
/**
|
|
* {@inheritdoc}
|
|
*
|
|
* @throws \Amp\Loop\AlreadyRunningException
|
|
*/
|
|
public function run() {
|
|
if ($this->running) {
|
|
throw new AlreadyRunningException("Cannot run loop recursively; loop already running");
|
|
}
|
|
|
|
$this->running = true;
|
|
|
|
try {
|
|
while ($this->running) {
|
|
if ($this->isEmpty()) {
|
|
return;
|
|
}
|
|
$this->tick();
|
|
}
|
|
} finally {
|
|
$this->stop();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* {@inheritdoc}
|
|
*/
|
|
public function stop() {
|
|
$this->running = false;
|
|
}
|
|
|
|
/**
|
|
* @return bool True if no referenced watchers remain in the loop.
|
|
*/
|
|
private function isEmpty() {
|
|
if (empty($this->watchers)) {
|
|
return true;
|
|
}
|
|
|
|
if (empty($this->unreferenced)) {
|
|
return false;
|
|
}
|
|
|
|
return \count($this->watchers) === \count($this->unreferenced);
|
|
}
|
|
|
|
/**
|
|
* Executes a single tick of the event loop.
|
|
*/
|
|
private function tick() {
|
|
try {
|
|
if (!empty($this->deferQueue)) {
|
|
$this->invokeDeferred();
|
|
}
|
|
|
|
$this->selectStreams($this->readStreams, $this->writeStreams, $this->getTimeout());
|
|
|
|
if (!empty($this->timerExpires)) {
|
|
$this->invokeTimers();
|
|
}
|
|
|
|
if ($this->signalHandling) {
|
|
\pcntl_signal_dispatch();
|
|
}
|
|
} catch (\Throwable $exception) {
|
|
if (null === $this->errorHandler) {
|
|
throw $exception;
|
|
}
|
|
|
|
$errorHandler = $this->errorHandler;
|
|
$errorHandler($exception);
|
|
} catch (\Exception $exception) { // @todo Remove when PHP 5.x support is no longer needed.
|
|
if (null === $this->errorHandler) {
|
|
throw $exception;
|
|
}
|
|
|
|
$errorHandler = $this->errorHandler;
|
|
$errorHandler($exception);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param resource[] $read
|
|
* @param resource[] $write
|
|
* @param int $timeout
|
|
*/
|
|
private function selectStreams(array $read, array $write, $timeout) {
|
|
$timeout /= self::MILLISEC_PER_SEC;
|
|
|
|
if (!empty($read) || !empty($write)) { // Use stream_select() if there are any streams in the loop.
|
|
if ($timeout >= 0) {
|
|
$seconds = (int) $timeout;
|
|
$microseconds = (int) (($timeout - $seconds) * self::MICROSEC_PER_SEC);
|
|
} else {
|
|
$seconds = null;
|
|
$microseconds = null;
|
|
}
|
|
|
|
$except = null;
|
|
|
|
// Error reporting suppressed since stream_select() emits an E_WARNING if it is interrupted by a signal.
|
|
$count = @\stream_select($read, $write, $except, $seconds, $microseconds);
|
|
|
|
if ($count) {
|
|
foreach ($read as $stream) {
|
|
$streamId = (int) $stream;
|
|
if (isset($this->readWatchers[$streamId])) {
|
|
foreach ($this->readWatchers[$streamId] as $watcher) {
|
|
$callback = $watcher->callback;
|
|
$callback($watcher->id, $stream, $watcher->data);
|
|
}
|
|
}
|
|
}
|
|
|
|
foreach ($write as $stream) {
|
|
$streamId = (int) $stream;
|
|
if (isset($this->writeWatchers[$streamId])) {
|
|
foreach ($this->writeWatchers[$streamId] as $watcher) {
|
|
$callback = $watcher->callback;
|
|
$callback($watcher->id, $stream, $watcher->data);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
if ($timeout > 0) { // Otherwise sleep with usleep() if $timeout > 0.
|
|
\usleep($timeout * self::MICROSEC_PER_SEC);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @return int Milliseconds until next timer expires or -1 if there are no pending times.
|
|
*/
|
|
private function getTimeout() {
|
|
while (!$this->timerQueue->isEmpty()) {
|
|
list($id, $timeout) = $this->timerQueue->top();
|
|
|
|
if (!isset($this->timerExpires[$id]) || $timeout !== $this->timerExpires[$id]) {
|
|
$this->timerQueue->extract(); // Timer was removed from queue.
|
|
continue;
|
|
}
|
|
|
|
$timeout -= (int) (\microtime(true) * self::MILLISEC_PER_SEC);
|
|
|
|
if ($timeout < 0) {
|
|
return 0;
|
|
}
|
|
|
|
return $timeout;
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
/**
|
|
* Invokes all pending defer watchers.
|
|
*/
|
|
private function invokeDeferred() {
|
|
$queue = $this->deferQueue;
|
|
|
|
foreach ($queue as $id) {
|
|
if (!isset($this->watchers[$id]) || !isset($this->deferQueue[$id])) {
|
|
continue;
|
|
}
|
|
|
|
$watcher = $this->watchers[$id];
|
|
unset($this->watchers[$id], $this->deferQueue[$id], $this->unreferenced[$id]);
|
|
|
|
$callback = $watcher->callback;
|
|
$callback($watcher->id, $watcher->data);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Invokes all pending delay and repeat watchers.
|
|
*/
|
|
private function invokeTimers() {
|
|
$time = (int) (\microtime(true) * self::MILLISEC_PER_SEC);
|
|
|
|
while (!$this->timerQueue->isEmpty()) {
|
|
list($id, $timeout) = $this->timerQueue->top();
|
|
|
|
if (!isset($this->timerExpires[$id]) || $timeout !== $this->timerExpires[$id]) {
|
|
$this->timerQueue->extract(); // Timer was removed from queue.
|
|
continue;
|
|
}
|
|
|
|
if ($this->timerExpires[$id] > $time) { // Timer at top of queue has not expired.
|
|
return;
|
|
}
|
|
|
|
// Remove and execute timer. Replace timer if persistent.
|
|
$this->timerQueue->extract();
|
|
|
|
$watcher = $this->watchers[$id];
|
|
|
|
if ($watcher->type === Watcher::REPEAT) {
|
|
$timeout = $time + $watcher->value;
|
|
$this->timerQueue->insert([$id, $timeout], -$timeout);
|
|
$this->timerExpires[$id] = $timeout;
|
|
} else {
|
|
unset($this->watchers[$id], $this->timerExpires[$id], $this->unreferenced[$id]);
|
|
}
|
|
|
|
// Execute the timer.
|
|
$callback = $watcher->callback;
|
|
$callback($watcher->id, $watcher->data);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* {@inheritdoc}
|
|
*/
|
|
public function defer(callable $callback, $data = null) {
|
|
$watcher = new Watcher;
|
|
$watcher->type = Watcher::DEFER;
|
|
$watcher->id = $this->nextId++;
|
|
$watcher->callback = $callback;
|
|
$watcher->data = $data;
|
|
|
|
$this->watchers[$watcher->id] = $watcher;
|
|
$this->deferQueue[$watcher->id] = $watcher->id;
|
|
|
|
return $watcher->id;
|
|
}
|
|
|
|
/**
|
|
* {@inheritdoc}
|
|
*/
|
|
public function delay($delay, callable $callback, $data = null) {
|
|
$delay = (int) $delay;
|
|
|
|
if ($delay <= 0) {
|
|
throw new \InvalidArgumentException("Delay must be greater than or equal to zero");
|
|
}
|
|
|
|
$watcher = new Watcher;
|
|
$watcher->type = Watcher::DELAY;
|
|
$watcher->id = $this->nextId++;
|
|
$watcher->callback = $callback;
|
|
$watcher->value = $delay;
|
|
$watcher->data = $data;
|
|
|
|
$this->watchers[$watcher->id] = $watcher;
|
|
|
|
$expiration = (int) (\microtime(true) * self::MILLISEC_PER_SEC) + $watcher->value;
|
|
|
|
$this->timerExpires[$watcher->id] = $expiration;
|
|
$this->timerQueue->insert([$watcher->id, $expiration], -$expiration);
|
|
|
|
return $watcher->id;
|
|
}
|
|
|
|
/**
|
|
* {@inheritdoc}
|
|
*/
|
|
public function repeat($interval, callable $callback, $data = null) {
|
|
$interval = (int) $interval;
|
|
|
|
if ($interval <= 0) {
|
|
throw new \InvalidArgumentException("Interval must be greater than or equal to zero");
|
|
}
|
|
|
|
$watcher = new Watcher;
|
|
$watcher->type = Watcher::REPEAT;
|
|
$watcher->id = $this->nextId++;
|
|
$watcher->callback = $callback;
|
|
$watcher->value = $interval;
|
|
$watcher->data = $data;
|
|
|
|
$this->watchers[$watcher->id] = $watcher;
|
|
|
|
$expiration = (int) (\microtime(true) * self::MILLISEC_PER_SEC) + $watcher->value;
|
|
|
|
$this->timerExpires[$watcher->id] = $expiration;
|
|
$this->timerQueue->insert([$watcher->id, $expiration], -$expiration);
|
|
|
|
return $watcher->id;
|
|
}
|
|
|
|
/**
|
|
* {@inheritdoc}
|
|
*/
|
|
public function onReadable($stream, callable $callback, $data = null) {
|
|
$watcher = new Watcher;
|
|
$watcher->type = Watcher::READABLE;
|
|
$watcher->id = $this->nextId++;
|
|
$watcher->callback = $callback;
|
|
$watcher->value = $stream;
|
|
$watcher->data = $data;
|
|
|
|
$this->watchers[$watcher->id] = $watcher;
|
|
$streamId = (int) $watcher->value;
|
|
$this->readWatchers[$streamId][$watcher->id] = $watcher;
|
|
$this->readStreams[$streamId] = $watcher->value;
|
|
|
|
return $watcher->id;
|
|
}
|
|
|
|
/**
|
|
* {@inheritdoc}
|
|
*/
|
|
public function onWritable($stream, callable $callback, $data = null) {
|
|
$watcher = new Watcher;
|
|
$watcher->type = Watcher::WRITABLE;
|
|
$watcher->id = $this->nextId++;
|
|
$watcher->callback = $callback;
|
|
$watcher->value = $stream;
|
|
$watcher->data = $data;
|
|
|
|
$this->watchers[$watcher->id] = $watcher;
|
|
$streamId = (int) $watcher->value;
|
|
$this->writeWatchers[$streamId][$watcher->id] = $watcher;
|
|
$this->writeStreams[$streamId] = $watcher->value;
|
|
|
|
return $watcher->id;
|
|
}
|
|
|
|
/**
|
|
* {@inheritdoc}
|
|
*
|
|
* @throws \Interop\Async\Loop\UnsupportedFeatureException If the pcntl extension is not available.
|
|
* @throws \Amp\Loop\SignalHandlerException If creating the backend signal handler fails.
|
|
*/
|
|
public function onSignal($signo, callable $callback, $data = null) {
|
|
if (!$this->signalHandling) {
|
|
throw new UnsupportedFeatureException("Signal handling requires the pcntl extension");
|
|
}
|
|
|
|
$watcher = new Watcher;
|
|
$watcher->type = Watcher::WRITABLE;
|
|
$watcher->id = $this->nextId++;
|
|
$watcher->callback = $callback;
|
|
$watcher->value = $signo;
|
|
$watcher->data = $data;
|
|
|
|
$this->enableSignal($watcher);
|
|
$this->watchers[$watcher->id] = $watcher;
|
|
|
|
return $watcher->id;
|
|
}
|
|
|
|
/**
|
|
* @param \Amp\Loop\Internal\Watcher $watcher
|
|
*
|
|
* @throws \Amp\Loop\SignalHandlerException If creating the backend signal handler fails.
|
|
*/
|
|
private function enableSignal(Watcher $watcher) {
|
|
if (!isset($this->signalWatchers[$watcher->value])) {
|
|
if (!@\pcntl_signal($watcher->value, function ($signo) {
|
|
foreach ($this->signalWatchers[$signo] as $watcher) {
|
|
if (!isset($this->watchers[$watcher->id])) {
|
|
continue;
|
|
}
|
|
|
|
$callback = $watcher->callback;
|
|
$callback($watcher->id, $signo, $watcher->data);
|
|
}
|
|
})) {
|
|
throw new SignalHandlerException("Failed to register signal handler");
|
|
}
|
|
}
|
|
|
|
$this->signalWatchers[$watcher->value][$watcher->id] = $watcher;
|
|
}
|
|
|
|
/**
|
|
* @param \Amp\Loop\Internal\Watcher $watcher
|
|
*/
|
|
private function disableSignal(Watcher $watcher) {
|
|
if (isset($this->signalWatchers[$watcher->value])) {
|
|
unset($this->signalWatchers[$watcher->value][$watcher->id]);
|
|
|
|
if (empty($this->signalWatchers[$watcher->value])) {
|
|
unset($this->signalWatchers[$watcher->value]);
|
|
@\pcntl_signal($watcher->value, \SIG_DFL);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* {@inheritdoc}
|
|
*/
|
|
public function setErrorHandler(callable $callback = null) {
|
|
$this->errorHandler = $callback;
|
|
}
|
|
|
|
/**
|
|
* {@inheritdoc}
|
|
*/
|
|
public function enable($watcherIdentifier) {
|
|
if (!isset($this->watchers[$watcherIdentifier])) {
|
|
throw new \LogicException("Cannot enable invalid watcher");
|
|
}
|
|
|
|
$watcher = $this->watchers[$watcherIdentifier];
|
|
|
|
switch ($watcher->type) {
|
|
case Watcher::READABLE:
|
|
$streamId = (int) $watcher->value;
|
|
$this->readWatchers[$streamId][$watcher->id] = $watcher->id;
|
|
$this->readStreams[$streamId] = $watcher->value;
|
|
break;
|
|
|
|
case Watcher::WRITABLE:
|
|
$streamId = (int) $watcher->value;
|
|
$this->writeWatchers[$streamId][$watcher->id] = $watcher->id;
|
|
$this->writeStreams[$streamId] = $watcher->value;
|
|
break;
|
|
|
|
case Watcher::DELAY:
|
|
case Watcher::REPEAT:
|
|
if (isset($this->timerExpires[$watcher->id])) {
|
|
break;
|
|
}
|
|
|
|
$expiration = (int) (\microtime(true) * self::MILLISEC_PER_SEC) + $watcher->value;
|
|
$this->timerExpires[$watcher->id] = $expiration;
|
|
$this->timerQueue->insert([$watcher->id, $expiration], -$expiration);
|
|
break;
|
|
|
|
case Watcher::DEFER:
|
|
$this->deferQueue[$watcher->id] = $watcher->id;
|
|
break;
|
|
|
|
case Watcher::SIGNAL:
|
|
$this->enableSignal($watcher);
|
|
break;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* {@inheritdoc}
|
|
*/
|
|
public function disable($watcherIdentifier) {
|
|
if (!isset($this->watchers[$watcherIdentifier])) {
|
|
return;
|
|
}
|
|
|
|
$watcher = $this->watchers[$watcherIdentifier];
|
|
|
|
switch ($watcher->type) {
|
|
case Watcher::READABLE:
|
|
$streamId = (int) $watcher->value;
|
|
unset($this->readWatchers[$streamId][$watcher->id]);
|
|
if (empty($this->readWatchers[$streamId])) {
|
|
unset($this->writeWatchers[$streamId], $this->readStreams[$streamId]);
|
|
}
|
|
break;
|
|
|
|
case Watcher::WRITABLE:
|
|
$streamId = (int) $watcher->value;
|
|
unset($this->writeWatchers[$streamId][$watcher->id]);
|
|
if (empty($this->writeWatchers[$streamId])) {
|
|
unset($this->writeWatchers[$streamId], $this->writeStreams[$streamId]);
|
|
}
|
|
break;
|
|
|
|
case Watcher::DELAY:
|
|
case Watcher::REPEAT:
|
|
unset($this->timerExpires[$watcher->id]);
|
|
break;
|
|
|
|
case Watcher::DEFER:
|
|
unset($this->deferQueue[$watcher->id]);
|
|
break;
|
|
|
|
case Watcher::SIGNAL:
|
|
$this->disableSignal($watcher);
|
|
break;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* {@inheritdoc}
|
|
*/
|
|
public function cancel($watcherIdentifier) {
|
|
$this->disable($watcherIdentifier);
|
|
unset($this->watchers[$watcherIdentifier], $this->unreferenced[$watcherIdentifier]);
|
|
}
|
|
|
|
/**
|
|
* {@inheritdoc}
|
|
*/
|
|
public function reference($watcherIdentifier) {
|
|
unset($this->unreferenced[$watcherIdentifier]);
|
|
}
|
|
|
|
/**
|
|
* {@inheritdoc}
|
|
*/
|
|
public function unreference($watcherIdentifier) {
|
|
if (!isset($this->watchers[$watcherIdentifier])) {
|
|
return;
|
|
}
|
|
|
|
$this->unreferenced[$watcherIdentifier] = $watcherIdentifier;
|
|
}
|
|
|
|
/**
|
|
* {@inheritdoc}
|
|
*/
|
|
public function info() {
|
|
$watchers = [
|
|
"referenced" => \count($this->watchers) - \count($this->unreferenced),
|
|
"unreferenced" => \count($this->unreferenced),
|
|
];
|
|
|
|
$defer = $delay = $repeat = $onReadable = $onWritable = $onSignal = [
|
|
"enabled" => 0,
|
|
"disabled" => 0,
|
|
];
|
|
|
|
foreach ($this->watchers as $watcher) {
|
|
switch ($watcher->type) {
|
|
case Watcher::READABLE:
|
|
if (isset($this->readWatchers[(int) $watcher->value][$watcher->id])) {
|
|
++$onReadable["enabled"];
|
|
} else {
|
|
++$onReadable["disabled"];
|
|
}
|
|
break;
|
|
|
|
case Watcher::WRITABLE:
|
|
if (isset($this->writeWatchers[(int) $watcher->value][$watcher->id])) {
|
|
++$onWritable["enabled"];
|
|
} else {
|
|
++$onWritable["disabled"];
|
|
}
|
|
break;
|
|
|
|
case Watcher::DEFER:
|
|
++$defer["enabled"];
|
|
break;
|
|
|
|
case Watcher::DELAY:
|
|
++$delay["enabled"];
|
|
break;
|
|
|
|
case Watcher::REPEAT:
|
|
if (isset($this->timerExpires[$watcher->id])) {
|
|
++$repeat["enabled"];
|
|
} else {
|
|
++$repeat["disabled"];
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
return [
|
|
"watchers" => $watchers,
|
|
"defer" => $defer,
|
|
"delay" => $delay,
|
|
"repeat" => $repeat,
|
|
"on_readable" => $onReadable,
|
|
"on_writable" => $onWritable,
|
|
"on_signal" => $onSignal,
|
|
"running" => $this->running,
|
|
];
|
|
}
|
|
|
|
/**
|
|
* {@inheritdoc}
|
|
*/
|
|
public function getHandle() {
|
|
return null;
|
|
}
|
|
|
|
/**
|
|
* Returns the same array of data as info().
|
|
*
|
|
* @return array
|
|
*/
|
|
public function __debugInfo() {
|
|
return $this->info();
|
|
}
|
|
} |