Dear Community and dear Go-developers,

Meanwhile it is clear why things do not work:
The Go-Scheduler is unable to allow to switch to another goroutine in 
busy-waiting-loops -
the only possibility to get around that problem is either to put 
"switch-steps" into the source
- either "time.Sleep(1)" or "runtime.Gosched()".
I think that THIS SHOULD BE DOCUMENTED IN THE LANGUAGE SPECIFICATION !!!!!

(See my book on https://maurer-berlin.eu/nspbuch - in it's 4th edition 
things are mentioned.)

With kind regards,
Christian Maurer

Am Montag, 30. Oktober 2017 18:30:06 UTC+1 schrieb Christian Maurer:
>
> Dear Go-community,
>
> I noticed a very strange effect by translating the
> mutual exclusion algorithm of E. W. Dijkstra to Go.
> Reference:
>   Cooperating Sequential Processes. Technical Report EWD-123,
>   Technological University Eindhoven (1965)
>   http://www.cs.utexas.edu/users/EWD/ewd01xx/EWD123.PDF)
>
> Inserting or omitting code lines A and B resp.
> changes the behaviour of this program very severely:
>
> With line A and without line B:
>   mutual exclusion is not guaranteed,
> with line B and without line A:
>   lock does not terminate and
> with both lines:
>   the program works as it should.
>
> --- 8< --------------------------------------------
> package main
>
> import (
>   "math/rand"
>   "time"
> )
>
> const N = 10 // number of goroutines involved
>
> var (
>   turn int
>   b, c = make([]int, N+1), make([]int, N+1)
>   n    = uint(0)
>   inCS = make([]bool, N+1)
>   s    = make([]string, N+1)
>   done = make(chan int)
> )
>
> func chk(i int) {
>   for j := 1; j <= N; j++ {
>     if j != i && inCS[j] {
>       panic("oops")
>     }
>   }
> }
>
> func lock(i int) {
>   b[i] = 0
> L:
>   if turn != i {
>     c[i] = 1
>     if b[turn] == 1 {
>       turn = i
>       goto L
>     }
>   }
>   time.Sleep(1) // A
>   c[i] = 0
>   time.Sleep(1) // B
>   for j := 1; j <= N; j++ {
>     if j != i && c[j] == 0 {
>       goto L
>     }
>   }
>   inCS[i] = true
>   chk(i)
> }
>
> func unlock(i int) {
>   inCS[i] = false
>   c[i], b[i] = 1, 1
>   //  turn = 0 // does not matter
> }
>
> func v() {
>   time.Sleep(time.Duration(1e7 + rand.Int63n(1e7)))
> }
>
> func count(i int) {
>   for k := 0; k < 100; k++ {
>     lock(i)
>     accu := n
>     v()
>     accu++
>     v()
>     n = accu
>     v()
>     println(s[i], n)
>     unlock(i)
>     v()
>   }
>   done <- 0
> }
>
> func main() {
>   s[1] = ""
>   for i := 2; i <= N; i++ {
>     s[i] = s[i-1] + "       "
>   }
>   for i := 1; i <= N; i++ {
>     go count(i)
>   }
>   for i := 1; i <= N; i++ {
>     <-done
>   }
> }
> --- >8 --------------------------------------------
>
> The cause for the "freaking out" of the program without
> those compulsory breaks (of only 1 ns) is absolutely unclear.
>
> Any sort of help is welcome!
>
> Kind regards to everybody,
>
> Christian Maurer
>
>

-- 
You received this message because you are subscribed to the Google Groups 
"golang-nuts" group.
To unsubscribe from this group and stop receiving emails from it, send an email 
to golang-nuts+unsubscr...@googlegroups.com.
To view this discussion on the web visit 
https://groups.google.com/d/msgid/golang-nuts/f83856a2-0661-4425-9861-196b83fcf07f%40googlegroups.com.

Reply via email to