1
0
mirror of https://github.com/danog/file.git synced 2025-01-23 05:41:18 +01:00
file/lib/Internal/EioPoll.php
Niklas Keller 6d2ca5e2b2 Fix EioPoll busy watcher
Previously we used listen() + done(). This API design is quite prone to
errors, as it's easy to miss a done() call in some branch. Additionally
this commit ensures that pending operations are always completed before
the EioPoll is completely destructed. Previously unfinished operations
leaked into new EioPoll instances.
2017-06-20 22:59:23 +02:00

76 lines
1.8 KiB
PHP

<?php
namespace Amp\File\Internal;
use Amp\CallableMaker;
use Amp\Loop;
use Amp\Promise;
class EioPoll {
use CallableMaker;
/** @var resource */
private static $stream;
/** @var string */
private $watcher;
/** @var int */
private $requests = 0;
/** @var callable */
private $onDone;
public function __construct() {
$this->onDone = $this->callableFromInstanceMethod("done");
if (!self::$stream) {
\eio_init();
self::$stream = \eio_get_event_stream();
}
$this->watcher = Loop::onReadable(self::$stream, static function () {
while (\eio_npending()) {
\eio_poll();
}
});
Loop::disable($this->watcher);
Loop::setState(self::class, new class ($this->watcher) {
private $watcher;
private $driver;
public function __construct(string $watcher) {
$this->watcher = $watcher;
$this->driver = Loop::get();
}
public function __destruct() {
$this->driver->cancel($this->watcher);
// Ensure there are no active operations anymore. This is a safe-guard as some operations might not be
// finished on loop exit due to not being yielded. This also ensures a clean shutdown for these if PHP
// exists.
\eio_event_loop();
}
});
}
public function listen(Promise $promise) {
if ($this->requests++ === 0) {
Loop::enable($this->watcher);
}
$promise->onResolve($this->onDone);
}
private function done() {
if (--$this->requests === 0) {
Loop::disable($this->watcher);
}
\assert($this->requests >= 0);
}
}