I don't think it needs to be uniformly fair. It just needs to prevent starvation of a channel for infinity. Any proof would have progress covered if we pick every channel once in a while, even under a heavily unfair weighting. You have to balance out the overhead of rolling a fair dice and the speed at which you can select on channels.
On Sat, Oct 28, 2017 at 12:54 AM Michael Jones <michael.jo...@gmail.com> wrote: > the bias of %n would not be detectable in this situation...but if you > wanted it right...the perfect thing is not hard. > > On Fri, Oct 27, 2017 at 3:44 PM, T L <tapir....@gmail.com> wrote: > >> Yes, the problem is fixed totally on tip. >> The tip is using the normal mod operation "fastrand() % n". >> >> >> On Friday, October 27, 2017 at 5:17:53 PM UTC-4, Axel Wagner wrote: >>> >>> It turns out that, apparently, at least on tip (what I linked to) the >>> fastrand implementation itself is actually pretty good. You might want to >>> re-run the test on tip. fastrandn will still be somewhat biased. >>> >>> On Fri, Oct 27, 2017 at 11:05 PM, Axel Wagner <axel.wa...@googlemail.com >>> > wrote: >>> >>>> Neither. fastrand is not a perfect PRNG, as that doesn't exist (much >>>> less in…fast) and the reduction is biased (necessarily, unless you are >>>> going to resort to rejection sampling, which is slow). >>>> >>>> On Fri, Oct 27, 2017 at 11:00 PM, T L <tapi...@gmail.com> wrote: >>>> >>>>> >>>>> >>>>> On Friday, October 27, 2017 at 4:57:46 PM UTC-4, T L wrote: >>>>>> >>>>>> >>>>>> >>>>>> On Friday, October 27, 2017 at 4:51:15 PM UTC-4, Axel Wagner wrote: >>>>>>> >>>>>>> Yes. The random number generator employed (right above from what I >>>>>>> linked to; the fastrand function), apparently, is unsurprisingly >>>>>>> imperfect. >>>>>>> >>>>>> >>>>>> Why not to maintain a global start index for each goroutine? >>>>>> The value of the start index is the number of executed select blocks >>>>>> in a goroutine. >>>>>> So that we can change the fastrandn implementation to >>>>>> "return uint32(uint64(fastrand() + current_g.numExecutedSelects) * >>>>>> uint64(n) >> 32)" >>>>>> >>>>> >>>>> Aha, which function is unfair, fastrand or fastrandn? >>>>> If fastrandn is unfair, then the above trick doesn't work. >>>>> >>>>>> >>>>>> >>>>>>> >>>>>>> On Fri, Oct 27, 2017 at 10:47 PM, T L <tapi...@gmail.com> wrote: >>>>>>> >>>>>>>> It is also not uniform for powers of two. >>>>>>>> If there a 8 cases, the sixth one will only get a 80% chance of the >>>>>>>> first one. >>>>>>>> >>>>>>>> On Friday, October 27, 2017 at 4:33:47 PM UTC-4, Axel Wagner wrote: >>>>>>>>> >>>>>>>>> The code suggests, that select is indeed not fair: >>>>>>>>> >>>>>>>>> https://github.com/golang/go/blob/a4d03a9bf7604b727abd0a1ebfb118ff6366ee50/src/runtime/stubs.go#L109 >>>>>>>>> Which seems fine to me. A fair, uniform PRNG for ranges that are >>>>>>>>> not powers of two gets a fair bit of overhead and no program should >>>>>>>>> depend >>>>>>>>> on uniformity of select for correctness anyway. >>>>>>>>> >>>>>>>>> On Fri, Oct 27, 2017 at 10:28 PM, T L <tapi...@gmail.com> wrote: >>>>>>>>> >>>>>>>>>> It looks the distribution of the RNG results depends on the >>>>>>>>>> number of cases. >>>>>>>>>> The third case only gets the lowest possibility of there are 6 >>>>>>>>>> cases, >>>>>>>>>> >>>>>>>>>> >>>>>>>>>> On Friday, October 27, 2017 at 4:21:12 PM UTC-4, T L wrote: >>>>>>>>>>> >>>>>>>>>>> A better example to show the inhomogeneity of the selected >>>>>>>>>>> possibilities of each case. >>>>>>>>>>> >>>>>>>>>>> package main >>>>>>>>>>> >>>>>>>>>>> import "fmt" >>>>>>>>>>> >>>>>>>>>>> func main() { >>>>>>>>>>> foo := make(chan struct{}) >>>>>>>>>>> close(foo) >>>>>>>>>>> var a, b, c, d, e, f, g float64 >>>>>>>>>>> fa := func(){a++} >>>>>>>>>>> fb := func(){b++} >>>>>>>>>>> fc := func(){c++} >>>>>>>>>>> fd := func(){d++} >>>>>>>>>>> fe := func(){e++} >>>>>>>>>>> ff := func(){f++} >>>>>>>>>>> fg := func(){g++} >>>>>>>>>>> for i := 0; i < 1000000; i++ { >>>>>>>>>>> select { >>>>>>>>>>> case <-foo: fa() >>>>>>>>>>> case <-foo: fb() >>>>>>>>>>> case <-foo: fc() >>>>>>>>>>> case <-foo: fd() >>>>>>>>>>> case <-foo: fe() >>>>>>>>>>> case <-foo: ff() >>>>>>>>>>> case <-foo: fg() >>>>>>>>>>> } >>>>>>>>>>> } >>>>>>>>>>> fmt.Println(a/a) >>>>>>>>>>> fmt.Println(b/a) >>>>>>>>>>> fmt.Println(c/a) >>>>>>>>>>> fmt.Println(d/a) >>>>>>>>>>> fmt.Println(e/a) >>>>>>>>>>> fmt.Println(f/a) >>>>>>>>>>> fmt.Println(g/a) >>>>>>>>>>> } >>>>>>>>>>> >>>>>>>>>>> result: >>>>>>>>>>> >>>>>>>>>>> 1 >>>>>>>>>>> 0.9999052991869258 >>>>>>>>>>> 0.9015652691532394 >>>>>>>>>>> 0.9691884140319548 >>>>>>>>>>> 0.966320332264567 >>>>>>>>>>> 0.9669020658305938 >>>>>>>>>>> 0.9604624105415533 >>>>>>>>>>> >>>>>>>>>>> On Friday, October 27, 2017 at 4:08:39 PM UTC-4, T L wrote: >>>>>>>>>>>> >>>>>>>>>>>> The following program constantly prints a value larger than 1.02 >>>>>>>>>>>> >>>>>>>>>>>> package main >>>>>>>>>>>> >>>>>>>>>>>> import "fmt" >>>>>>>>>>>> >>>>>>>>>>>> func main() { >>>>>>>>>>>> foo, bar := make(chan struct{}), make(chan struct{}) >>>>>>>>>>>> close(foo) >>>>>>>>>>>> close(bar) >>>>>>>>>>>> x, y := 0.0, 0.0 >>>>>>>>>>>> f := func(){x++} >>>>>>>>>>>> g := func(){y++} >>>>>>>>>>>> for i := 0; i < 1000000; i++ { >>>>>>>>>>>> select { >>>>>>>>>>>> case <-bar: g() >>>>>>>>>>>> case <-foo: f() >>>>>>>>>>>> case <-bar: g() >>>>>>>>>>>> case <-foo: f() >>>>>>>>>>>> case <-bar: g() >>>>>>>>>>>> case <-foo: f() >>>>>>>>>>>> } >>>>>>>>>>>> } >>>>>>>>>>>> fmt.Println(x/y) >>>>>>>>>>>> } >>>>>>>>>>>> >>>>>>>>>>>> if the last two cases are removed, then it becomes quit fair. >>>>>>>>>>>> >>>>>>>>>>> -- >>>>>>>>>> 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...@googlegroups.com. >>>>>>>>>> For more options, visit https://groups.google.com/d/optout. >>>>>>>>>> >>>>>>>>> >>>>>>>>> -- >>>>>>>> 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...@googlegroups.com. >>>>>>>> For more options, visit https://groups.google.com/d/optout. >>>>>>>> >>>>>>> >>>>>>> -- >>>>> 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...@googlegroups.com. >>>>> For more options, visit https://groups.google.com/d/optout. >>>>> >>>> >>>> >>> -- >> 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. >> For more options, visit https://groups.google.com/d/optout. >> > > > > -- > Michael T. Jones > michael.jo...@gmail.com > > -- > 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. > For more options, visit https://groups.google.com/d/optout. > -- 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. For more options, visit https://groups.google.com/d/optout.