2016-12-30 02:16:04 +01:00
|
|
|
<?php
|
2015-08-10 05:30:11 +02:00
|
|
|
|
2016-08-23 23:47:40 +02:00
|
|
|
namespace Amp\Parallel\Threading\Internal;
|
2016-08-18 18:04:48 +02:00
|
|
|
|
2016-08-23 23:47:40 +02:00
|
|
|
use Amp\{ Coroutine, Pause };
|
|
|
|
use Amp\Parallel\Sync\Lock;
|
2017-01-09 18:11:25 +01:00
|
|
|
use AsyncInterop\Promise;
|
2015-08-10 05:30:11 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
2016-08-18 18:04:48 +02:00
|
|
|
class Mutex extends \Threaded {
|
|
|
|
const LATENCY_TIMEOUT = 10;
|
2015-08-10 05:30:11 +02:00
|
|
|
|
2016-08-26 17:10:03 +02:00
|
|
|
/** @var bool */
|
2015-08-10 05:30:11 +02:00
|
|
|
private $lock = true;
|
2016-08-18 18:04:48 +02:00
|
|
|
|
|
|
|
/**
|
2017-01-09 18:11:25 +01:00
|
|
|
* @return \AsyncInterop\Promise
|
2016-08-18 18:04:48 +02:00
|
|
|
*/
|
2016-11-15 00:43:44 +01:00
|
|
|
public function acquire(): Promise {
|
2016-08-18 18:04:48 +02:00
|
|
|
return new Coroutine($this->doAcquire());
|
|
|
|
}
|
|
|
|
|
2015-08-10 05:30:11 +02:00
|
|
|
/**
|
|
|
|
* Attempts to acquire the lock and sleeps for a time if the lock could not be acquired.
|
|
|
|
*
|
|
|
|
* @return \Generator
|
|
|
|
*/
|
2016-08-18 18:04:48 +02:00
|
|
|
public function doAcquire(): \Generator {
|
2015-08-10 05:30:11 +02:00
|
|
|
$tsl = function () {
|
|
|
|
return ($this->lock ? $this->lock = false : true);
|
|
|
|
};
|
|
|
|
|
2015-09-02 03:58:22 +02:00
|
|
|
while (!$this->lock || $this->synchronized($tsl)) {
|
2016-08-18 18:04:48 +02:00
|
|
|
yield new Pause(self::LATENCY_TIMEOUT);
|
2015-08-10 05:30:11 +02:00
|
|
|
}
|
|
|
|
|
2016-01-23 07:00:56 +01:00
|
|
|
return new Lock(function () {
|
2015-08-10 05:30:11 +02:00
|
|
|
$this->release();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Releases the lock.
|
|
|
|
*/
|
2016-08-18 18:04:48 +02:00
|
|
|
protected function release() {
|
2015-08-10 05:30:11 +02:00
|
|
|
$this->lock = true;
|
|
|
|
}
|
2015-08-28 21:41:27 +02:00
|
|
|
}
|