forked from acl2/acl2
-
Notifications
You must be signed in to change notification settings - Fork 0
/
multi-threading-raw.lisp
1094 lines (885 loc) · 42.8 KB
/
multi-threading-raw.lisp
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
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
; ACL2 Version 8.1 -- A Computational Logic for Applicative Common Lisp
; Copyright (C) 2018, Regents of the University of Texas
; This version of ACL2 is a descendent of ACL2 Version 1.9, Copyright
; (C) 1997 Computational Logic, Inc. See the documentation topic NOTE-2-0.
; This program is free software; you can redistribute it and/or modify
; it under the terms of the LICENSE file distributed with ACL2.
; 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
; LICENSE for more details.
; Written by: Matt Kaufmann and J Strother Moore
; email: [email protected] and [email protected]
; Department of Computer Science
; University of Texas at Austin
; Austin, TX 78712 U.S.A.
; We thank David L. Rager for contributing an initial version of this file.
; This file is divided into the following sections.
; Section: Enabling and Disabling Interrupts
; Section: Multi-Threading Interface
; Section: Multi-Threading Utility Functions and Macros
; Section: Multi-Threading Constants
(in-package "ACL2")
; For readability we use #+sb-thread" instead of #+(and sbcl sbl-thread). We
; therefore make the following check to ensure that these two readtime
; conditionals are equivalent.
#+(and (not sbcl) sb-thread)
(error "Feature sb-thread not supported on Lisps other than SBCL")
; We would like to use the semaphore notification objects supported in SBCL
; 1.0.54+, but version 1.0.54 has a heap exhaustion error that we wish to
; avoid. As such, for now, we remain compatible with 1.0.53 and below. While
; we can consider uncommenting the following feature push once SBCL's support
; for semaphore notification objects is stable, the following questions seem
; difficult to answer: How can we determine when that is the case, and how
; confident are we that Rager's implementation using semaphore notification
; objects is correct? Rager determined that 1.0.54 was unstable by making all
; of the books with parallelism enabled (and experiencing the "heap exhausted"
; error in one of the ordinal books [at the moment of this writing, he does not
; recall which one]). Thus, Rager's code has not been completely tested using
; semaphore notification objects.
; #+sbcl
; (push :sbcl-sno-patch *features*)
;---------------------------------------------------------------------
; Section: Enabling and Disabling Interrupts
; Without-interrupts and unwind-protect-disable-interrupts-during-cleanup were
; originally defined here, but have been moved to acl2-fns.lisp in order to
; support not only ACL2(p) but also ACL2(h).
;---------------------------------------------------------------------
; Section: Threading Interface
;
; The threading interface is intended for system level programmers. It is not
; intended for the ACL2 user. When writing system-level multi-threaded code,
; we use implementation-independent interfaces. If you need a function not
; covered in this interface, create the interface!
; Many of the functions in this interface (lockp, make-lock, and so on) are not
; used elsewhere, but are included here in case we find a use for them later.
; We take a conservative approach for implementations that do not support
; parallelism. For example, if the programmer asks for a semaphore or lock in
; an unsupported Lisp, then nil is returned.
; We employ counting semaphores. For details, including a discussion of
; ordering, see comments in the definition of function make-semaphore.
; Note: We use parts of the threading interface for our implementation of the
; parallelism primitives.
#+sb-thread
(defstruct (atomically-modifiable-counter
(:constructor make-atomically-modifiable-counter-raw))
; SBCL has an atomic increment and decrement interface. In all small tests,
; this interface works well. However, when we use this interface to
; parallelize the waterfall of ACL2, it doesn't work. The observable bug that
; we find is that our Fixnum number is decremented too much (so it becomes -1,
; which is a really large non-negative fixnum). Rather than sort out the cause
; of this bug, we simply revert to using locks in SBCL. However, we leave the
; code that implements the use of these atomic increments and decrements. This
; achieves two things: (1) The code is there if we decide we want to use it
; again later (maybe after reading on the SBCL email lists that a bug with
; these atomic operations was fixed) and (2) We will have an on-going record of
; the code that causes us to observe this bug (note that ACL2 4.2 also
; contains the implementation that uses atomic operations). Maybe someone will
; be able to tell us what we are doing wrong.
(val 0 ; :type (unsigned-byte #+x86-64 64 #-x86-64 32)
)
(lock (sb-thread:make-mutex :name "counter lock")))
(defun make-atomically-modifiable-counter (initial-value)
#+ccl
initial-value
#+sb-thread
(make-atomically-modifiable-counter-raw :val initial-value)
#+lispworks
initial-value
#-(or ccl sb-thread lispworks)
initial-value)
(defmacro define-atomically-modifiable-counter (name initial-value)
`(defvar ,name (make-atomically-modifiable-counter ,initial-value)))
(defmacro atomically-modifiable-counter-read (counter)
#+ccl
counter
#+sb-thread
`(atomically-modifiable-counter-val ,counter)
#+lispworks
counter
#-(or ccl sb-thread lispworks)
counter)
(defmacro atomic-incf (x)
; Warning: CCL and SBCL return different values for atomic-incf. As of Oct
; 2009, CCL returns the new value, but SBCL returns the old value. We
; artificially add one to the SBCL return value to make them consistent. Both
; the CCL maintainer Gary Byers and the SBCL community have confirmed the
; return value of atomic-incf/decf to be reliable.
#+ccl
`(ccl::atomic-incf ,x)
#+sb-thread
; Parallelism blemish: we (David Rager) performed experiments in either 2009 or
; 2010 that indicated that there was a problem with using SBCL's built-in
; atomic increment and decrements. Because of this observation, we roll our
; own atomic increments and decrements based on locking. For future reference,
; we leave the version of the code that used SBCL's built-in primitives as a
; comment.
; `(progn ; (sb-debug:backtrace)
; (1+ (sb-ext:atomic-incf
; (atomically-modifiable-counter-val ,x))))
; Parallelism blemish: we do not recall why nil appears in the call to
; with-recursive-lock below. Probably it can be omitted.
`(sb-thread:with-recursive-lock ((atomically-modifiable-counter-lock ,x))
nil ; see comment above
(incf (atomically-modifiable-counter-val ,x)))
#+lispworks
`(system:atomic-incf ,x)
#-(or ccl sb-thread lispworks)
`(incf ,x))
(defmacro atomic-incf-multiple (counter count)
; Warning: CCL and SBCL return different values for atomic-incf. As of Oct
; 2009, CCL returns the new value, but SBCL returns the old value. We
; artificially add one to the SBCL return value to make them consistent. Both
; the CCL maintainer Gary Byers and the SBCL community have confirmed the
; return value of atomic-incf/decf to be reliable. According to the Lispworks
; documentation, Lispworks returns the new value.
#+ccl
`(without-interrupts
; Admittedly, the following loop can be viewed as not being "atomic". But the
; important thing is that two threads don't increment from the same value, and
; the use of atomic-incf guarantees that.
(dotimes (i ,count) (ccl::atomic-incf ,counter)))
#+sb-thread
; `(+ (sb-ext:atomic-incf
; (atomically-modifiable-counter-val ,counter) ,count)
; ,count)
`(sb-thread:with-recursive-lock ((atomically-modifiable-counter-lock ,counter)) nil
(incf (atomically-modifiable-counter-val ,counter) ,count))
#+lispworks
`(system:atomic-incf ,counter ,count)
#-(or ccl sb-thread lispworks)
`(incf ,counter ,count))
(defmacro atomic-decf (x)
; Warning: CCL and SBCL return different values for atomic-incf. As of Oct
; 2009, CCL returns the new value, but SBCL returns the old value. We
; artificially subtract one from the SBCL return value to make them consistent.
; Both the CCL maintainer Gary Byers and the SBCL community have confirmed the
; return value of atomic-incf/decf to be reliable.
#+ccl
`(ccl::atomic-decf ,x)
#+sb-thread
; `(let ((ret-val (1- (sb-ext::atomic-decf (atomically-modifiable-counter-val
; ,x) 1))))
; (when (> ret-val 999999999999)
; (print "Warning: resetting atomically modifiable counter to 0")
; (setf ,x (make-atomically-modifiable-counter-raw :val 0)))
; 0)
`(sb-thread:with-recursive-lock ((atomically-modifiable-counter-lock ,x)) nil
(decf (atomically-modifiable-counter-val ,x)))
#+lispworks
`(system:atomic-decf ,x)
#-(or ccl sb-thread lispworks)
`(decf ,x))
(defun lockp (x)
#+ccl (cl:typep x 'ccl::recursive-lock)
#+sb-thread (cl:typep x 'sb-thread::mutex)
#+lispworks (cl:typep x 'mp:lock)
#-(or ccl sb-thread lispworks)
; We return nil in the uni-threaded case in order to stay in sync with
; make-lock, which returns nil in this case. In a sense, we want (lockp
; (make-lock x)) to be a theorem if there is no error.
(null x))
; Make-lock and with-lock were originally defined in this file, but has been
; moved to acl2-fns.lisp to help support multi-threading in memoize-raw.lisp.
(defmacro reset-lock (bound-symbol)
; This macro binds the given global (but not necessarily special) variable to a
; lock that is new, at least from a programmer's perspective.
; Reset-lock should only be applied to bound-symbol if deflock has previously
; been applied to bound-symbol.
`(setq ,bound-symbol (make-lock ,(symbol-name bound-symbol))))
(defun run-thread (name fn-symbol &rest args)
; Apply fn-symbol to args. We follow the precedent set by LISP machines (and
; in turn CCL), which allowed the user to spawn a thread whose initial
; function receives an arbitrary number of arguments.
; We expect this application to occur in a fresh thread with the given name.
; When a call of this function returns, we imagine that this fresh thread can
; be garbage collected; at any rate, we don't hang on to it!
; Note that run-thread returns different types in different Lisps.
; A by-product of our use of lambdas is that fn-symbol doesn't have to be a
; function symbol. It's quite fine to call run-thread with a lambda, e.g.
;
; (run-thread "hello" (lambda () (print "hi")))
;
; A more sophisticated version of run-thread would probably check whether
; fn-symbol was indeed a symbol and only create a new lambda if it was.
#-(or ccl sb-thread lispworks)
(declare (ignore name))
#+ccl
(ccl:process-run-function name (lambda () (apply fn-symbol args)))
#+sb-thread
(sb-thread:make-thread (lambda () (apply fn-symbol args)) :name name)
#+lispworks
(mp:process-run-function name nil (lambda () (apply fn-symbol args)))
; We're going to be nice and let the user's function still run, even though
; it's not split off.
#-(or ccl sb-thread lispworks)
(apply fn-symbol args))
(defun interrupt-thread (thread function &rest args)
; Interrupt the indicated thread and then, in that thread, apply function to
; args. Note that function and args are all evaluated. When this function
; application returns, the thread resumes from the interrupt (from where it
; left off).
#-(or ccl sb-thread lispworks)
(declare (ignore thread function args))
#+ccl
(apply #'ccl:process-interrupt thread function args)
#+sb-thread
(if args
(error "Passing arguments to interrupt-thread not supported in SBCL.")
(sb-thread:interrupt-thread thread function))
#+lispworks
(apply #'mp:process-interrupt thread function args)
#-(or ccl sb-thread lispworks)
nil)
(defun kill-thread (thread)
#-(or ccl sb-thread lispworks)
(declare (ignore thread))
#+ccl
(ccl:process-kill thread)
#+sb-thread
(sb-thread:terminate-thread thread)
#+lispworks
(mp:process-kill thread)
#-(or ccl sb-thread lispworks)
nil)
(defun all-threads ()
#+ccl
(ccl:all-processes)
#+sb-thread
(sb-thread:list-all-threads)
#+lispworks
(mp:list-all-processes)
#-(or ccl sb-thread lispworks)
(error "We don't know how to list threads in this Lisp."))
(defun current-thread ()
#+ccl
ccl:*current-process*
#+sb-thread
sb-thread:*current-thread*
#+lispworks
mp:*current-process*
#-(or ccl sb-thread lispworks)
nil)
(defun thread-wait (fn &rest args)
; Thread-wait provides an inefficient mechanism for the current thread to wait
; until a given condition, defined by the application of fn to args, is true.
; When performance matters, we advise using a signaling mechanism instead of
; this relatively highly-latent function.
#+ccl
(apply #'ccl:process-wait "Asynchronously waiting on a condition" fn args)
#+lispworks
(apply #'mp:process-wait "Asynchronously waiting on a condition" fn args)
#-(or ccl lispworks)
(loop while (not (apply fn args)) do (sleep 0.05)))
#+(or sb-thread lispworks)
(defmacro with-potential-timeout (body &key timeout)
; There is no implicit progn for the body argument. This is different from
; sb-sys:with-deadline, but we figure the simplicity is more valuable than
; randomly passing in a :timeout value.
; Note that if there is a timeout, the return value is unspecified.
#+sb-thread
`(if ,timeout
(handler-case
(sb-sys:with-deadline
(:seconds ,timeout)
,body)
(sb-ext:timeout ()))
,body)
#+lispworks
(lispworks:with-unique-names (process timer)
`(catch 'lispworks-timeout
(let* ((,process mp:*current-process*)
(,timer (mp:make-timer (lambda ()
(mp:process-interrupt
,process
(lambda ()
(throw 'lispworks-timeout nil)))))))
(unwind-protect-disable-interrupts-during-cleanup
(progn
(mp:schedule-timer-relative ,timer ,timeout)
,body)
(mp:unschedule-timer ,timer))))))
; We would like to find a clean way to provide the user with an implicit progn,
; while still maintaining timeout as a keyword argument.
; #+sb-thread
; (defmacro with-potential-sbcl-timeout (&rest body &key timeout)
;
; ; The below use of labels is only necessary because we provide an implicit
; ; progn for the body of with-potential-sbcl-timeout.
;
; (let ((correct-body
; (labels ((remove-keyword-from-list
; (lst keyword)
; (if (or (atom lst) (atom (cdr lst)))
; lst
; (if (equal (car lst) :timeout)
; (cddr lst)
; (cons (car lst) (remove-keyword-from-args (cdr lst)))))))
; (remove-keyword-from-args body :timeout))))
;
;
; `(if ,timeout
; (handler-case
; (sb-sys:with-deadline
; (:seconds ,timeout)
; ,@correct-body)
;
; (sb-ext:timeout ()))
; ,@correct-body)))
; Essay on Condition Variables
; A condition variable is a data structure that can be passed to corresponding
; "wait" and "signal" functions. When a thread calls the wait function on a
; condition variable, c, the thread blocks until "receiving a signal" from the
; application of the signal function to c. Only one signal is sent per call of
; the signal function; so, at most one thread will unblock. (There is a third
; notion for condition variable, namely the broadcast function, which is like
; the signal function except that all threads blocking on the given condition
; variable will unblock. But we do not support broadcast functions in this
; interface, in part because we use semaphores for CCL, and there's no way
; to broadcast when you're really using a semaphore.)
; The design of our parallelism library is simpler when using condition
; variables for the following reason: Since a worker must wait for two
; conditions before consuming work, it is better to use a condition variable
; and test those two conditions upon waking, rather than try and use two
; semaphores.
; Implementation Note: As of March 2007, our CCL implementation does not
; yield true condition variables. A condition variable degrades to a
; semaphore, so if one thread first signals a condition variable, then that
; signal has been stored. Then later (perhaps much later), when another thread
; waits for that signal, that thread will be able to proceed by decrementing
; the count. As a result the later thread will "receive" the signal, even
; though that signal occurred in the past. Fortunately, this isn't a
; contradiction of the semantics of condition variables, since with condition
; variables there is no specification of how far into the future the waiting
; thread will receive a signal from the signaling thread.
; Note: Condition variables should not be used to store state. They are only a
; signaling mechanism, and any state update implied by receiving a condition
; variable's signal should be checked. This usage is believed to be consistent
; with traditional condition variable semantics.
(defun make-condition-variable ()
; If CCL implements condition variables, we will want to change the CCL
; expansion and remove the implementation note above.
; Because implementing broadcast for condition variables in CCL is much more
; heavyweight than a simple semaphore, we keep it simple until we have a use
; case for a broadcast. Such simple requirements are satisfied by using a
; semaphore.
#+ccl
(ccl:make-semaphore)
#+sb-thread
(sb-thread:make-waitqueue)
#+lispworks
(mp:make-condition-variable)
#-(or ccl sb-thread lispworks)
nil)
(defmacro signal-condition-variable (cv)
#-(or ccl sb-thread lispworks)
(declare (ignore cv))
#+ccl
`(ccl:signal-semaphore ,cv)
#+sb-thread
; According to an email sent by Gabor Melis, of SBCL help, on 2007-02-25, if
; there are two threads waiting on a condition variable, and a third thread
; signals the condition variable twice before either can receive the signal,
; then both threads should receive the signal. If only one thread unblocks, it
; is considered a bug.
`(sb-thread:condition-notify ,cv)
#+lispworks
`(mp:condition-variable-signal ,cv)
; Parallelism wart: we would like for signal-condition-variable to return nil
; to signify that this macro is only evaluated for side-effects. However, it
; appears that this macro's return value is actually needed, because the
; theorem prover hangs consistently in SBCL and LispWorks when we do return nil
; here. As such, we currently remove the returning of nil. But, this is a
; hotfix. A much better solution is to change the reliance upon this macro's
; return value and then uncomment the following nil.
; nil
)
(defmacro broadcast-condition-variable (cv)
#-(or sb-thread lispworks)
(declare (ignore cv))
#+ccl
(error "Broadcasting condition variables is unsupported in CCL")
#+sb-thread
`(sb-thread:condition-broadcast ,cv)
#+lispworks
`(mp:condition-variable-broadcast ,cv)
nil)
(defun wait-on-condition-variable (cv lock &key timeout)
; This is only executed for side-effect.
; A precondition to this function is that the current thread "owns" lock. This
; is a well-known part of how condition variables work. This is also
; documented in the SBCL manual in section 12.5 entitled "Waitqueue/condition
; variables."
; Parallelism blemish: when timeout is provided, the return value of t can be
; misleading. This isn't a huge problem, because condition variables are only
; used for signaling, and so, assuming the programmer is following the
; discipline associated with using condition variables, any thread waiting on a
; condition variable should check the state of the program for the property it
; needs anyway. And, if that property is not provided, the thread will just
; rewait on the condition variable. So, the only real penalty should be a
; slight loss of performance. Of course, this small penalty depends on the
; following the discipline of using condition variables correctly. We may not
; wish to rely upon the programmer's ability to follow this discipline, so we
; leave it as a parallelism blemish.
#-(or sb-thread lispworks)
(declare (ignore cv lock timeout))
#+ccl
(error "Waiting on a condition variable is unsupported in CCL")
#+sb-thread
(with-potential-timeout
(sb-thread:condition-wait cv lock)
:timeout timeout)
#+lispworks
(mp:condition-variable-wait cv lock :timeout timeout)
t)
#+sb-thread
(defstruct acl2-semaphore
(lock (sb-thread:make-mutex))
(cv (sb-thread:make-waitqueue)) ; condition variable
(count 0))
#+lispworks
(defstruct acl2-semaphore
(lock (mp:make-lock))
(cv (mp:make-condition-variable)) ; condition variable
(count 0))
(defun make-semaphore (&optional name)
; Make-semaphore, signal-semaphore, and semaphorep work together to implement
; counting semaphores for the threading interface.
; This function creates "counting semaphores", which are data structures that
; include a "count" field, which is a natural number. A thread can "wait on" a
; counting semaphore, and it will block in the case that the semaphore's count
; is 0. To "signal" such a semaphore means to increment that field and to
; notify a unique waiting thread (we will discuss a relaxation of this
; uniqueness shortly) that the semaphore's count has been incremented. Then
; this thread, which is said to "receive" the signal, decrements the
; semaphore's count and is then unblocked. This mechanism is typically much
; faster than busy waiting.
; In principle more than one waiting thread could be notified (though this
; seems rare in practice). In this case, only one would be the receiving
; thread, i.e., the one that decrements the semaphore's count and is then
; unblocked.
; If semaphore usage seems to perform inefficiently, could this be due to
; ordering issues? For example, even though CCL nearly always uses a FIFO
; for blocked threads, it does not guarantee so: no such promise is made by the
; CCL documentation or implementor. Thus, we make absolutely no guarantees
; about ordering; for example, we do not guarantee that the longest-blocked
; thread for a given semaphore is the one that would receive a signal.
; However, we suspect that this will usually be the case for most
; implementations, so assuming such an ordering property is probably a
; reasonable heuristic. We would be somewhat surprised to find significant
; performance problems in our own application to ACL2's parallelism primitives
; due to the ordering provided by the underlying system.
; CCL provides us with semaphores for signaling. SBCL provides condition
; variables for signaling. Since we want to code for one type of signaling
; between parents and children, we create a semaphore wrapper for SBCL's
; condition variables. The structure sbcl-semaphore implements the data for
; this wrapper.
; Followup: SBCL has recently (as of November 2010) implemented semaphores, and
; the parallelism code could be changed to reflect this. However, since SBCL
; does not implement semaphore-notification-object's, we choose to stick with our
; own implementation of semaphores for now.
(declare (ignore name))
#+ccl
(ccl:make-semaphore)
#+(or (and sb-thread (not sbcl-sno-patch)) lispworks)
(make-acl2-semaphore)
#+(and sb-thread sbcl-sno-patch)
(sb-thread:make-semaphore)
#-(or ccl sb-thread lispworks)
; We return nil in the uni-threaded case in order to stay in sync with
; semaphorep.
nil)
(defun semaphorep (semaphore)
; Make-semaphore, signal-semaphore, and semaphorep work together to implement
; counting semaphores for our threading interface.
; This function recognizes our notion of semaphore structures.
#+ccl
(typep semaphore 'ccl::semaphore)
#+sb-thread
(and (acl2-semaphore-p semaphore)
(typep (acl2-semaphore-lock semaphore) 'sb-thread::mutex)
(typep (acl2-semaphore-cv semaphore) 'sb-thread::waitqueue)
(integerp (acl2-semaphore-count semaphore)))
#+lispworks
(and (acl2-semaphore-p semaphore)
(typep (acl2-semaphore-lock semaphore) 'mp::lock)
(typep (acl2-semaphore-cv semaphore) 'mp::condition-variable)
(integerp (acl2-semaphore-count semaphore)))
#-(or ccl sb-thread lispworks)
; We return nil in the uni-threaded case in order to stay in sync with
; make-semaphore, which returns nil in this case. In a sense, we want
; (semaphorep (make-semaphore x)) to be a theorem if there is no error.
(null semaphore))
(defun make-semaphore-notification ()
; This function returns an object that records when a corresponding semaphore
; has been signaled (for use when wait-on-semaphore is called with that
; semaphore and that object).
#+ccl
(ccl:make-semaphore-notification)
#+(or (and sb-thread (not sbcl-sno-patch)) lispworks)
(make-array 1 :initial-element nil)
#+(and sb-thread sbcl-sno-patch)
(sb-thread:make-semaphore-notification)
#-(or ccl sb-thread lispworks)
nil)
(defun semaphore-notification-status (semaphore-notification-object)
#-(or ccl sb-thread lispworks)
(declare (ignore semaphore-notification-object))
#+ccl
(ccl:semaphore-notification-status semaphore-notification-object)
#+(or (and sb-thread (not sbcl-sno-patch)) lispworks)
(aref semaphore-notification-object 0)
#+(and sb-thread sbcl-sno-patch)
(sb-thread:semaphore-notification-status semaphore-notification-object)
#-(or ccl sb-thread lispworks)
; t may be the wrong default, but we don't have a use case for this return
; value yet, so we postpone thinking about the "right" value until we are aware
; of a need.
t)
(defun clear-semaphore-notification-status (semaphore-notification-object)
#-(or ccl sb-thread lispworks)
(declare (ignore semaphore-notification-object))
#+ccl
(ccl:clear-semaphore-notification-status semaphore-notification-object)
#+(or (and sb-thread (not sbcl-sno-patch)) lispworks)
(setf (aref semaphore-notification-object 0) nil)
#+(and sb-thread sbcl-sno-patch)
(sb-thread:clear-semaphore-notification semaphore-notification-object)
#-(or ccl sb-thread lispworks)
nil)
; We implement this only for SBCL and Lispworks, because even a system-level
; programmer is not expected to use this function. We use it only from within
; the threading interface to implement wait-on-semaphore for SBCL and
; Lispworks.
(defun set-semaphore-notification-status (semaphore-notification-object)
#-(or sb-thread lispworks)
(declare (ignore semaphore-notification-object))
#+(or sb-thread lispworks)
(setf (aref semaphore-notification-object 0) t)
#-(or sb-thread lispworks)
(error
"Set-semaphore-notification-status not supported outside SBCL or Lispworks"))
(defun signal-semaphore (semaphore)
; Make-semaphore, signal-semaphore, and semaphorep work together to implement
; counting semaphores for our threading interface.
; This function is executed for side effect; the value returned is irrelevant.
#-(or ccl sb-thread lispworks)
(declare (ignore semaphore))
#+ccl
(ccl:signal-semaphore semaphore)
#+(or (and sb-thread (not sbcl-sno-patch)) lispworks)
(with-lock-raw
(acl2-semaphore-lock semaphore)
(without-interrupts
(incf (acl2-semaphore-count semaphore))
(signal-condition-variable (acl2-semaphore-cv semaphore))))
#+(and sb-thread sbcl-sno-patch)
(sb-thread:signal-semaphore semaphore)
nil)
; Once upon a time, we optimized the manual allocation and deallocation of
; semaphores so that they could be recycled. CCL and SBCL have since evolved,
; and as such, we have removed the implementation code and its corresponding
; uses.
(defun wait-on-semaphore (semaphore &key notification timeout)
; This function is guaranteed to return t when there is no timeout provided and
; the function call received the signal. Its return value when the signal has
; not been received is unspecified. As such, we provide the semaphore
; notification object as a means for determining whether a signal was actually
; received.
; Parallelism blemish: if a timeout is included, then it is our intent that a
; call of this function that times-out should return nil. This is known to be
; the case in CCL, but while we have written the code for SBCL and LispWorks,
; we have not actually tested the code for this property.
; This function only returns normally after receiving a signal for the given
; semaphore, setting the notification status of notification (if supplied and
; non-nil) to true; see semaphore-notification-status. But control can leave
; this function abnormally, for example if the thread executing a call of this
; function is interrupted (e.g., with interface function interrupt-thread) with
; code that does a throw, in which case notification is unmodified.
; We need the ability to know whether we received a signal or not. CCL
; provides this through a semaphore notification object. SBCL does not provide
; this mechanism currently, so we might "unreceive the signal" in the cleanup
; form of the implementation. We do this by only decrementing the count of the
; semaphore iff we set the notification object. This means we have to resignal
; the semaphore if we were interrupted while signaling, but we would have to do
; this anyway.
#-(or ccl sb-thread lispworks)
(declare (ignore semaphore notification timeout))
#+ccl
(if timeout
(ccl:timed-wait-on-semaphore semaphore timeout notification)
(progn (ccl:wait-on-semaphore semaphore notification)
t))
#+(or (and sb-thread (not sbcl-sno-patch)) lispworks)
(let ((received-signal nil))
; If we did not use a variable like "received-signal", we could have the
; following race condition:
; -- Suppose Thread A waits for the semaphore and is waiting for the
; signal from the CV
; -- Suppose Thread B is in the same state, that is, waiting for the
; semaphore and for the signal from the CV
; -- Then thread C signals the CV
; -- Thread A is awakened by the signal but is immediately interrupted
; by another thread and forced to throw a tag which is caught higher up
; in Thread A's call stack.
; -- The signal is effectively "lost", so while the semaphore count may
; be accurate, Thread B is still waiting on a signal that will never
; come.
; We guard against this lost signal by always re-signaling the condition
; variable (unless we're sure we received the signal, which we would know
; because "received-signal" would be set to t).
; This signaling during the unwind portion of the unwind protect definitely
; results in some inefficient execution. This is brought about because now any
; thread that waits on a signal will automatically signal the condition
; variable any time that it doesn't receive the signal (either due to a timeout
; or an interrupt+throw/error). However, this is the price of liveness for our
; system (which requires we implement semaphores in user space because we need
; semaphore-notification-objects).
(with-lock-raw
(acl2-semaphore-lock semaphore)
(unwind-protect-disable-interrupts-during-cleanup
(with-potential-timeout
(progn
(loop while (<= (acl2-semaphore-count semaphore) 0) do
; The current thread missed the chance to decrement and must rewait. This can
; only occur if another thread grabbed the lock and decremented the
(wait-on-condition-variable (acl2-semaphore-cv semaphore)
(acl2-semaphore-lock semaphore)))
(setq received-signal t)
t) ; if this progn returns, this t is the return value
:timeout timeout)
(if received-signal
; The current thread was able to record the receipt of the signal. The current
; thread will decrement the count of the semaphore and set the semaphore
; notification object.
(progn
(decf (acl2-semaphore-count semaphore))
(when notification
(set-semaphore-notification-status notification)))
; The current thread may have received the signal but been unable to record it.
; In this case, the current thread will signal the condition variable again, so
; that any other thread waiting on the semaphore can have a chance at acquiring
; the said semaphore. This results in needlessly signaling the condition
; variable portion of the semaphore every time a timeout occurs. However, that
; is the cost of a "live" implementation ("live" loosely means "makes progress").
(signal-condition-variable (acl2-semaphore-cv semaphore)))))
(if timeout received-signal t))
#+(and sb-thread sbcl-sno-patch)
(if timeout
(and (sb-thread:wait-on-semaphore semaphore
:timeout timeout
:notification notification)
t)
(progn (sb-thread:wait-on-semaphore semaphore
:timeout timeout
:notification notification)
t))
#-(or ccl sb-thread lispworks)
t) ; default is to receive a semaphore/lock
;---------------------------------------------------------------------
; Section: Multi-Threading Utility Functions and Macros
;
; These functions and macros could be defined in parallel-raw.lisp, except that
; we also use them in futures-raw.lisp. Rather than create another file, we
; place them here, at the end of the multi-threading interface.
(defvar *throwable-worker-thread*
; When we terminate threads due to a break and abort, we need a way to
; terminate all threads. We implement this by having them throw the
; :worker-thread-no-longer-needed tag. Unfortunately, sometimes the threads
; are outside the scope of the associated catch, when throwing the tag would
; cause an error. We avoid this warning by maintaining the dynamically-bound
; variable *throwable-worker-thread*. When the throwable context is entered,
; we let a new copy of the variable into existence and set it to T. Now, when
; we throw :worker-thread-no-longer-needed, we only throw it if
; *throwable-worker-thread* is non-nil.
nil)
(defun throw-all-threads-in-list (thread-list)
; We interrupt each of the given threads with a throw to the catch at the top
; of consume-work-on-work-queue-when-there, which is the function called
; by run-thread in spawn-worker-threads-if-needed.
; Compare with kill-all-threads-in-list, which kills all of the given threads
; (typically all user-produced threads), not just those self-identified as
; being within the associated catch block.
(if (endp thread-list)
nil
(progn
#-sbcl
(interrupt-thread
(car thread-list)
#'(lambda () (when *throwable-worker-thread*
(throw :worker-thread-no-longer-needed nil))))
#+sbcl
(handler-case
(interrupt-thread
(car thread-list)
#'(lambda () (when *throwable-worker-thread*
(throw :worker-thread-no-longer-needed nil))))
(sb-thread:interrupt-thread-error
()
; Parallelism no-fix: it turns out that this error is common. And since we
; think that the error is caused by the thread being finished by the time we
; get around to interrupting it, we do not print the following warning. The
; error message we are trying to avoid is:
; debugger invoked on a SB-THREAD:INTERRUPT-THREAD-ERROR in thread #<THREAD
; "initial thread" RUNNING
; {100E761FB1}>:
; Interrupt thread failed: thread #<THREAD "Worker thread" FINISHED values: 17
; {100F1AD651}> has exited.
; (format t "Warning: We tried to interrupt a thread and throw it with ~%~
; the :worker-thread-no-longer-needed tag, but we ~%~
; encountered an error. Since the error is likely benign, ~%~
; this is only a warning. Unless you are an ACL2(p) ~%~
; maintainer, you can ignore this warning.~%")
))
(throw-all-threads-in-list (cdr thread-list)))))
(defun kill-all-threads-in-list (thread-list)
; Compare with throw-all-threads-in-list, which uses throw instead of killing
; threads directly, but only affects threads self identified as being within an
; associated catch block.
(if (endp thread-list)
nil
(progn
(kill-thread (car thread-list))
(kill-all-threads-in-list (cdr thread-list)))))
(defun thread-name (thread)
#+ccl
(ccl:process-name thread)
#+sbcl
(sb-thread:thread-name thread)
#+lispworks
(mp:process-name thread)
#-(or ccl sbcl lispworks)
(error "Function thread-name is unimplemented in this Lisp."))
(defconstant *worker-thread-name* "Worker thread")
(defun worker-threads1 (threads)
(cond ((endp threads)
nil)
((equal (thread-name (car threads)) *worker-thread-name*)
(cons (car threads)
(worker-threads1 (cdr threads))))
(t (worker-threads1 (cdr threads)))))
(defun worker-threads ()
; Returns the subset of the current list of threads that are worker threads
; spawned by ACL2(p).
(worker-threads1 (all-threads)))
(defun all-worker-threads-are-dead ()
#+sbcl
(<= (length (all-threads)) 1)
#-sbcl
(null (worker-threads)))
#+ccl
(defun all-given-threads-are-reset (threads)
(if (endp threads)
t
(and (equal (ccl:process-whostate (car threads))
"Reset")
(all-given-threads-are-reset (cdr threads)))))
(defun all-worker-threads-are-dead-or-reset ()
; CCL has a bug that if you try to throw a thread that is "reset" ("reset" is a
; CCL term), it ignores the throw (and the thread never expires). As such, we
; do not let threads in the "reset" state prevent us from finishing the
; resetting of parallelism variables.
#+ccl
(all-given-threads-are-reset (worker-threads))
#-ccl
(null (worker-threads)))
(defun send-die-to-worker-threads ()
; This function is evaluated only for side effect.
(throw-all-threads-in-list (worker-threads))
(let ((round 0))
(loop do
; We used to call "(thread-wait 'all-worker-threads-are-dead)".
; However, we noticed a synchronization problem between what we might prefer
; the underlying Lisp to do (in this one case) and what the Lisp actually does.
; In particular, when we call run-thread, the call to run-thread returns,
; before we know that the thread has actually started running. This is
; probably fine in the general case. However, in this instance, it is possible
; for the threads that are about to be run to not receive the throw, because
; they haven't really started running yet. Rather than work out the details of
; this problem, we punt, and simply wait for one second before issuing a new
; throw. In practice, this works just fine.
(when (equal (mod round 10) 0)
(throw-all-threads-in-list (worker-threads))
; The following commented code is only for debugging
; (format t "Waiting for parallelism worker threads to halt.~%")
; (format t "Current threads are ~%~s~%~%" (all-threads))
)
(sleep 0.03)
(incf round)
while (not (all-worker-threads-are-dead-or-reset)))))
(defun kill-all-worker-threads ()
; This function is evaluated only for side effect.
(kill-all-threads-in-list (worker-threads))
(thread-wait 'all-worker-threads-are-dead))