File: mvar.sml

package info (click to toggle)
smlsharp 4.1.0-1
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid, trixie
  • size: 123,732 kB
  • sloc: ansic: 16,725; sh: 4,347; makefile: 2,191; java: 742; haskell: 493; ruby: 305; cpp: 284; pascal: 256; ml: 255; lisp: 141; asm: 97; sql: 74
file content (59 lines) | stat: -rw-r--r-- 1,493 bytes parent folder | download | duplicates (2)
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
(**
 * mvar.sml
 *
 * @copyright (C) 2021 SML# Development Team.
 * @author UENO Katsuhiro
 *)

structure MVar :> sig

  type 'a mvar
  val new : unit -> 'a mvar
  val put : 'a mvar * 'a -> unit
  val take : 'a mvar -> 'a

end =
struct

  type 'a mvar =
       {mutex: Pthread.pthread_mutex_t,
        cond: Pthread.pthread_cond_t,
        content: 'a option ref}

  fun new () =
      let
        val mutex = Pthread.new_pthread_mutex_t ()
        val _ = Pthread.pthread_mutex_init (mutex, Pointer.NULL ())
        val cond = Pthread.new_pthread_cond_t ()
        val _ = Pthread.pthread_cond_init (cond, Pointer.NULL ())
      in
        {mutex = mutex, cond = cond, content = ref NONE} : 'a mvar
      end

  fun waitUntil f (mvar as {mutex, cond, content}:'a mvar) =
      if f (!content) then ()
      else (Pthread.pthread_cond_wait (cond, mutex); waitUntil f mvar)

  fun put (mvar as {mutex, cond, content}:'a mvar, value) =
      (
        Pthread.pthread_mutex_lock mutex;
        waitUntil (not o isSome) mvar;
        content := SOME value;
        Pthread.pthread_cond_signal cond;
        Pthread.pthread_mutex_unlock mutex;
        ()
      )

  fun take (mvar as {mutex, cond, content}:'a mvar) =
      let
        val _ = Pthread.pthread_mutex_lock mutex
        val _ = waitUntil isSome mvar
        val ret = valOf (!content)
      in
        content := NONE;
        Pthread.pthread_cond_signal cond;
        Pthread.pthread_mutex_unlock mutex;
        ret
      end

end