2 * Copyright (C) 2003-2004:
3 * Stefano Zacchiroli <zack@cs.unibo.it>
4 * for the HELM Team http://helm.cs.unibo.it/
6 * This file is part of HELM, an Hypertextual, Electronic
7 * Library of Mathematics, developed at the Computer Science
8 * Department, University of Bologna, Italy.
10 * HELM is free software; you can redistribute it and/or
11 * modify it under the terms of the GNU General Public License
12 * as published by the Free Software Foundation; either version 2
13 * of the License, or (at your option) any later version.
15 * HELM is distributed in the hope that it will be useful,
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 * GNU General Public License for more details.
20 * You should have received a copy of the GNU General Public License
21 * along with HELM; if not, write to the Free Software
22 * Foundation, Inc., 59 Temple Place - Suite 330, Boston,
25 * For details, see the HELM World-Wide-Web page,
26 * http://helm.cs.unibo.it/
32 let debug_print s = if debug then prerr_endline (Lazy.force s)
37 val mutex = Mutex.create ()
39 (** condition variable: 'no readers is currently reading' *)
40 val noReaders = Condition.create ()
43 val mutable readersCount = 0
45 method private incrReadersCount = (* internal, not exported *)
46 self#doCritical (lazy (
47 readersCount <- readersCount + 1
50 method private decrReadersCount = (* internal, not exported *)
51 self#doCritical (lazy (
52 if readersCount > 0 then readersCount <- readersCount - 1;
55 method private signalNoReaders = (* internal, not exported *)
56 self#doCritical (lazy (
57 if readersCount = 0 then Condition.signal noReaders
60 method private doCritical: 'a. 'a lazy_t -> 'a =
62 debug_print (lazy "<doCritical>");
65 let res = Lazy.force action in
67 debug_print (lazy "</doCritical>");
73 method private doReader: 'a. 'a lazy_t -> 'a =
75 debug_print (lazy "<doReader>");
77 self#decrReadersCount;
80 self#incrReadersCount;
81 let res = (try Lazy.force action with e -> (cleanup (); raise e)) in
83 debug_print (lazy "</doReader>");
86 (* TODO may starve!!!! is what we want or not? *)
87 method private doWriter: 'a. 'a lazy_t -> 'a =
89 debug_print (lazy "<doWriter>");
90 self#doCritical (lazy (
91 while readersCount > 0 do
92 Condition.wait noReaders mutex
94 let res = Lazy.force action in
95 debug_print (lazy "</doWriter>");