blob: 6356903f4ded87d2c03dfb42b992b074b0c5dd76 (
plain)
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
|
# Commands covered: (test)mutex
#
# This file contains a collection of tests for one or more of the Tcl
# built-in commands. Sourcing this file into Tcl runs the tests and
# generates output for errors. No output means no errors were found.
#
# Copyright (c) 2025 Ashok P. Nadkarni
#
# See the file "license.terms" for information on usage and redistribution
# of this file, and for a DISCLAIMER OF ALL WARRANTIES.
if {"::tcltest" ni [namespace children]} {
package require tcltest 2.5
namespace import -force ::tcltest::*
}
source [file join [file dirname [info script]] tcltests.tcl]
::tcltest::loadTestedCommands
package require -exact tcl::test [info patchlevel]
testConstraint testmutex [expr {[info commands testmutex] ne {}}]
namespace eval testmutex {
namespace import ::tcltest::test
proc testlock {id nthreads recursion iters yield} {
test $id "mutex lock $nthreads/$recursion/$iters/$yield" \
-constraints testmutex \
-body "testmutex lock $nthreads $recursion $iters $yield" \
-result [expr {$nthreads*$iters}]
}
# threads recursions iterations yield
testlock mutex-lock-1 2 1 1000000 0
testlock mutex-lock-2 2 1 1000000 1
testlock mutex-lock-3 10 1 200000 0
testlock mutex-lock-4 10 1 200000 1
testlock mutex-lock-5 4 5 400000 0
testlock mutex-lock-6 4 5 400000 1
proc fairness {totalOps perThreadOps} {
set errors {}
set threadTotal [tcl::mathop::+ {*}$perThreadOps]
if {$threadTotal ne $totalOps} {
append errors "Thread total $threadTotal != expected $totalOps\n"
}
# Each thread should get at least half of fair share
set fairShare [expr {$totalOps / [llength $perThreadOps]}]
foreach share $perThreadOps {
if {$fairShare > 2*$share} {
append errors "Thread share $share < 0.5 fair share $fairShare"
}
}
return $errors
}
proc testcondition {id nthreads recursion iters yield} {
set totalOps [expr {$nthreads*$iters}]
test $id "mutex condition $nthreads/$recursion/$iters/$yield" \
-constraints testmutex \
-body {
lassign \
[testmutex condition $nthreads $recursion $iters $yield] \
enqTotal enqPerThread enqTimeouts \
deqTotal deqPerThread deqTimeouts
list \
$enqTotal [fairness $totalOps $enqPerThread] $enqTimeouts \
$deqTotal [fairness $totalOps $deqPerThread] $deqTimeouts
} -result [list $totalOps {} 0 $totalOps {} 0]
}
testcondition mutex-condition-1 2 1 100000 0
testcondition mutex-condition-2 2 1 100000 1
testcondition mutex-condition-3 10 1 20000 0
testcondition mutex-condition-4 10 1 20000 1
testcondition mutex-condition-5 4 5 40000 0
testcondition mutex-condition-6 4 5 40000 1
}
# cleanup
::tcltest::cleanupTests
return
|