This file is indexed.

/usr/lib/ocaml/stm.mli is in libcothreads-ocaml-dev 0.10-3build5.

This file is owned by root:root, with mode 0o644.

The actual contents of the file can be viewed below.

  1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
(***********************************************************************)
(*                                                                     *)
(*                   STM library for OCaml                             *)
(*                                                                     *)
(*           (C) 2007 by Zheng Li (li@pps.jussieu.fr)                  *)
(*                                                                     *)
(*  This program is free software; you can redistribute it and/or      *)
(*  modify it under the terms of the GNU Library General Public        *)
(*  License as published by the Free Software Foundation; either       *)
(*  version 2 of the License, or (at your option) any later version.   *)
(*                                                                     *)
(*  This program is distributed in the hope that it will be useful,    *)
(*  but WITHOUT ANY WARRANTY; without even the implied warranty of     *)
(*  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the      *)
(*  GNU Library General Public License for more details.               *)
(*                                                                     *)
(***********************************************************************)

(** A user-space STM library for vmthreads, system threads, process and network
    programs(TODO).
*)


(** {6 STM model} *)


type 'a stm
  (** The type of a transaction, when executed will produce a result of type
  ['a]. *) 

val return: 'a -> 'a stm
  (** Primitive to wrap a plain of type ['a] value to a ['a stm], which when
      being executed, will produces the orignal value. *)

val bind: 'a stm -> ('a -> 'b stm) -> 'b stm
  (** [bind t f] is a transaction, when executed, first behavior as
      transaction [t], then feed the reture value to [f] to get the
      consecutive transaction to execute next.
  *)

val ( >>= ): 'a stm -> ('a -> 'b stm) -> 'b stm
  (** [t >>= f] is an alternative notation of [bind t f] *)

val ( >> ): 'a stm -> 'b stm -> 'b stm
  (** [t1 >> t2] is equal to [t1 >>= fun _ -> t2] which first execute [t1] and
      wait for its result but ignore it, and then behaviors like [t2]
  *)

val abort: 'a stm
  (** [abort] is a single transaction, when executed, abort the whole execution
      from current point. The result of abort an execution is detailed in the
      documentation of execution primitive [atom_once] below.
  *)

val retry: 'a stm
  (** [retry] is a transaction, when executed, first wait for the changing of
      any transactional variables being read in the history of current
      execution, then relaunch the whole execution.
  *)

val retry_now: 'a stm
  (** [retry_now] is a transaction in the same spirit with [retry], the only
      difference is that it does not wait for any changes and relaunch the
      execution immediately. This can be useful in a language with side-effect,
      see also the documentation on [atom_once]
  *)

val wait: unit stm
  (** [wait] is a transaction, when executed, simply wait for the changing of
      any transactional variables being read in the history of current
      execution, but without relaunch it. Semantically, you can consider [retry]
      as [wait >> retry_now]
  *)

val or_else: 'a stm -> 'a stm -> 'a stm
  (** [or_else t1 t2] is a transaction, when executed, first try to execute
      [t1]. If not encountering any [retry] ([retry_now]) or [abort], it
      behaviors just as [t1], otherwise it try [t2] in the same way. If both
      [t1] and [t2] [abort], the whole execution will abort; if either of them
      [retry_now], the whole execution will relaunch immediately; if either of
      them [retry], the execution will wait on its waiting condition and then
      relaunch; if both of them retry, the execution will wait on both their
      waiting conditions and then relaunch.
  *)

val catch: 'a stm -> (exn -> 'a stm) -> 'a stm
  (** [catch t f] is a transaction, when executed, behaviors as [t] if no
      exception arise, otherwise [f] is used to catch this exception and
      produce the replacing transaction to execute. Note that [catch] is
      transaction-level [try ... with].

      E.g., [let t3 = t1 >>= fun x -> let y = possible_exn x in t2], here [t3] is
      a transaction with potential exception inside,  [try t3 with ...] won't
      catch it because t3 is not executed yet, and [try atom_once t3 with ...]
      can catch it but then [t3] lost its composability. The solution is [catch
      t3 (function _ -> t4)] which catches exceptions {i inside} [t3] and
      results a valid transaction being able to furtherly composed; another
      possibility is to catch the exception normally {i outside} transactions
      like [t1 >>= fun x -> let y = try possible_exn x with _ -> some_value
      in t2]. 
  *)

val atom_once: 'a stm -> 'a option 
  (** [atom_once] execute a transaction and result in [Some v] if the
      transaction success and [None] if the transaction fail (due to
      conflicting in committing or abort). One difference between OCaml and
      Haskell is that OCaml is not pure and can hide side-effect anywhere while
      Haskell is pure and can seperate values with/without side-effect by
      types. On STM, any transaction may fail and relaunch for some times
      before its success, so any side-effects inside the transaction may be
      launched several times. Haskell forbid side effect inside transaction
      through types, whereas we won't be able to do that with OCaml. Instead of
      asking, but have no means to detect or forbid, the programmers to program
      without side-effect inside transaction, or modifying heavily the
      underlying run-time of a language with a imperative nature to be able to
      catch/revert side-effect, we simply tell it from type that "A transaction
      {i may} fail" and let the programmer decide what to do. This kind of
      things already exist in OCaml such as exception. E.g. [try incr i;
      danger_v1 with _ -> decr i; v2], it's the programmers' responsibility to
      revert [i] or choose not to do the side-effect modification inside a
      dangerous envrionment, if that's what they mean. On the other hand, the good 
      thing is that now the programmers have more flexibility in controlling the
      execution of transactions, e.g. they may choose in purpose not to
      repeatedly execute the transaction after the committing fails [x] times.
  *)

val atom: 'a stm -> 'a
  (** This is an analog of [atomically] in Haskell, which repeatedly execute
      a transaction until the committing succeed. As being said in [atom_once],
      the control is given to the programmer, the [atom] can defined by themselves
      as [let rec atom t = match atom_once t with Some v -> v | _ -> atom t]. 
      Providing it is just for convenience. In the same way, you can define various
      helper functions such as [check].

      As already warned (see [atom_once], transactions may fail, relauching 
      transactions also means relaunching side-effects inside a transaction if any.
      So usually you should avoid side-effect, unless it's something you don't care
      or even something you want: as an example, you may want to add a harmless print
      routine inside the transaction to be able to debug that how may times the
      transaction fails before its success :)

      Unlike in haskell, we allow nested [atom_once] or [atom].
  *)


(** {6 Transactional variable} *)

type 'a tvar
  (** the type of transactional variable, which has inside a value of type ['a] *)

val tvar: 'a -> 'a tvar
  (** Toplevel tvar declaration, produce a transaction variable from a
      value. See [new_tvar] *)

val new_tvar: 'a -> 'a tvar stm
  (** We provide two functions to create a transactional variable from common
      value: [tvar] is traditional toplevel declaration as those new* and
      create* functions seen in most other library, it is ensured to succeed;
      while [new_tvar] is a transactional declaration (as in Haskell) which may
      fail if the execution of the whole transaction it's bound in fails.

      We do not follow the idea on the relation between tvar allocation
      and exception from the original STM paper (last paragraph of section
      3.5), as we explicitly provide this two different declaration method:
      toplevel declaration [tvar] is ensured to be succeed; [new_tvar] is
      itself transactional, hence by no means should it be specially ensured to
      succeed always, if it's value is exposed after a fail transaction by
      exception etc, visiting it later will result in a Not_found exception,
      which exactly indicates what has happened: the [new_tvar] does not
      succeed.
  *)

val read_tvar: 'a tvar -> 'a stm
  (** Read value from a transactional variable, results in a transaction which
      can be further composed with other transactions through [bind] etc., or
      executed right away with [atom] etc. to get the final result
  *)

val write_tvar: 'a tvar -> 'a -> unit stm
  (** [write_tvar tv v] write value [v] to transactional variable [tv], results
      in a transaction whose type is [unit]. As [read_tvar], the result
      transaction is for composing or executing.

      Warning: do not operate the value of a transactional variable though other
      way exception write_tvar: such as producing the tvar from a mutable value
      or reference and secretly changing it in traditional way. First, it
      breaks the transactional semantics; second, in all possibility you won't
      be able to do that, as the value of tvar is isolated.
  *)