sof3/rwlock

Exclusive and Read-write locks for PHP

0.1.1 2022-03-05 13:06 UTC

This package is auto-updated.

Last update: 2024-12-28 04:11:52 UTC


README

Exclusive and Read-write locks for PHP.

This is a quick port of my javascript library rwlock-promise to PHP. Some parts may happen to be less idiomatic because of this.

This library uses the await-generator framework to expose asynchronous function call API.

Usage

$mutex = new Mutex;

$epoch = time();

yield $mutex->run(function() use($epoch) {
	echo "Start 1: ", time() - $epoch, "\n";
	yield $this->waitSeconds(2);
	echo "End 1: ", time() - $epoch, "\n";
});

yield $mutex->run(function() use($epoch) {
	echo "Start 2: ", time() - $epoch, "\n";
	yield $this->waitSeconds(1);
	echo "End 2: ", time() - $epoch, "\n";
});

The above should output

Start 1: 0
End 1: 2
Start 2: 2
End 2: 3

The second generator is only run after the first generator returns.

$mutex = new RwLock;

$epoch = time();

Await::g2c($mutex->readClosure(function() use($epoch) {
	echo "Start 1: ", time() - $epoch, "\n";
	yield $this->waitSeconds(2);
	echo "End 1: ", time() - $epoch, "\n";
}));

Await::g2c($mutex->readClosure(function() use($epoch, $mutex) {
	echo "Start 2: ", time() - $epoch, "\n";
	yield $this->waitSeconds(1);
	echo "End 2: ", time() - $epoch, "\n";

	Await::g2c($mutex->readClosure(function() use($epoch) {
		echo "Start 3: ", time() - $epoch, "\n";
		yield $this->waitSeconds(2);
		echo "End 3: ", time() - $epoch, "\n";
	}));

	Await::f2c($mutex->writeClosure(function() use($epoch) {
		echo "Start 4: ", time() - $epoch, "\n";
		yield $this->waitSeconds(2);
		echo "End 4: ", time() - $epoch, "\n";
	}));

	Await::f2c($mutex->readClosure(function() use($epoch) {
		echo "Start 5: ", time() - $epoch, "\n";
		yield $this->waitSeconds(2);
		echo "End 5: ", time() - $epoch, "\n";
	}));
}));

The above should output

Start 1: 0
Start 2: 0
End 2: 1
Start 3: 1
End 1: 2
End 3: 3
Start 4: 3
End 4: 5
Start 5: 5
End 5: 7

It does not make sense to yield (await) a read/write inside a read/write block; in particular, yielding another write inside a read/write block would lead to a deadlock. Use Await::g2c, as shown above, to schedule a read/write without blocking on it. However, you must always either yield or f2c/g2c a read/write, because it only returns a generator, which does nothing if you don't call anything on it.