This repository has been archived on 2023-08-20. You can view files and clone it, but cannot push or open issues or pull requests.
yap-6.3/Logtalk/examples/threads/philosophers/philosophers.lgt

236 lines
4.6 KiB
Plaintext
Raw Normal View History

:- category(chopstick).
:- info([
version is 1.1,
author is 'Paulo Moura',
date is 2006/12/2,
comment is 'Dining philosophers problem: chopstick representation.']).
:- public(pick_up/0).
:- mode(pick_up, zero_or_one).
:- info(pick_up/0, [
comment is 'A Philosopher picks up the chopstick.']).
:- public(put_down/0).
:- mode(put_down, zero_or_one).
:- info(put_down/0, [
comment is 'A Philosopher puts down the chopstick.']).
:- private(available/0).
:- dynamic(available/0).
:- mode(available, zero_or_one).
:- info(available/0, [
comment is 'Chopstick state (either available or in use).']).
% chopstick actions (picking up and putting down) are synchronized using the same mutext
% such that a chopstick can only be handled by a single philosopher at a time:
:- synchronized([pick_up/0, put_down/0]).
pick_up :-
::available,
::retract(available).
put_down :-
\+ ::available,
::asserta(available).
:- end_category.
:- object(cs1,
imports(chopstick)).
:- dynamic(available/0).
available.
:- end_object.
:- object(cs2,
imports(chopstick)).
:- dynamic(available/0).
available.
:- end_object.
:- object(cs3,
imports(chopstick)).
:- dynamic(available/0).
available.
:- end_object.
:- object(cs4,
imports(chopstick)).
:- dynamic(available/0).
available.
:- end_object.
:- object(cs5,
imports(chopstick)).
:- dynamic(available/0).
available.
:- end_object.
:- category(philosopher).
:- info([
version is 1.0,
author is 'Paulo Moura',
date is 2006/12/2,
comment is 'Dining philosophers problem: philosopher representation.']).
:- public(left_chopstick/1).
:- mode(left_chopstick(?object_identifier), zero_or_one).
:- info(left_chopstick/1, [
comment is 'Chopstick at the left of a philosopher.',
argnames is ['Chopstick']]).
:- public(right_chopstick/1).
:- mode(right_chopstick(?object_identifier), zero_or_one).
:- info(right_chopstick/1, [
comment is 'Chopstick at the right of a philosopher.',
argnames is ['Chopstick']]).
:- public(run/2).
:- mode(run(+integer, +integer), one).
:- info(run/2, [
comment is 'Runs Count number of thinking/eating cycles, with each activity taking MaxTime (in seconds).',
argnames is ['Count', 'MaxTime']]).
:- private(message/1).
:- synchronized(message/1).
:- mode(message(+list), one).
:- info(message/1, [
comment is 'Writes all the terms on a list as an atomic operation.',
argnames is ['Atoms']]).
:- private(random/2).
:- synchronized(random/2).
:- mode(random(+integer, -integer), one).
:- info(random/2, [
comment is 'Ensures synchronized access to the random number generator.',
argnames is ['Limit', 'Random']]).
run(0, _) :-
this(Philosopher),
message([Philosopher, ' terminated.']).
run(Count, MaxTime) :-
think(MaxTime),
( eat(MaxTime) ->
Count2 is Count - 1,
run(Count2, MaxTime)
; run(Count, MaxTime)
).
think(MaxTime):-
this(Philosopher),
random(MaxTime, ThinkTime),
message(['Philosopher ', Philosopher, ' thinking for ', ThinkTime, ' seconds.']),
sleep(ThinkTime).
% deadlock while a philosopher is trying to eat is prevented by putting
% down the first chopstick when picking up the second one fails:
eat(MaxTime):-
this(Philosopher),
random(MaxTime, EatTime),
::left_chopstick(LeftStick),
::right_chopstick(RightStick),
LeftStick::pick_up,
( RightStick::pick_up ->
message(['Philosopher ', Philosopher, ' eating for ', EatTime, ' seconds with chopsticks ', LeftStick, ' and ', RightStick, '.']),
sleep(EatTime),
::LeftStick::put_down,
::RightStick::put_down
; ::LeftStick::put_down,
fail
).
% as the "random" library object is not multi-threading aware, we must use a
% synchronized wrap up predicate (random/2) to call the random number generator:
random(Limit, Value) :-
random::random(1, Limit, Value).
% writing a message needs to be synchronized as it's accomplished
% using a combination of individual write/1 (and nl/0) calls:
message([]) :-
nl,
flush_output.
message([Atom| Atoms]) :-
write(Atom),
message(Atoms).
:- end_category.
:- object(p1,
imports(philosopher)).
:- threaded.
left_chopstick(cs5).
right_chopstick(cs1).
:- end_object.
:- object(p2,
imports(philosopher)).
:- threaded.
left_chopstick(cs1).
right_chopstick(cs2).
:- end_object.
:- object(p3,
imports(philosopher)).
:- threaded.
left_chopstick(cs3).
right_chopstick(cs2).
:- end_object.
:- object(p4,
imports(philosopher)).
:- threaded.
left_chopstick(cs4).
right_chopstick(cs3).
:- end_object.
:- object(p5,
imports(philosopher)).
:- threaded.
left_chopstick(cs5).
right_chopstick(cs4).
:- end_object.