Version 11 of sleep

Updated 2004-08-04 18:35:15

Why does Tcl not come with a sleep command? Why, it just doesn't call it sleep. It calls it after.

However, Jeff Hobbs notes in a discussion held elsewhere:

[someone proposed the code ]

  proc sleep {time} {
      after $time set end 1
      vwait end
  }

[and Jeff responds:]

This [the above code] is actually not correct because vwait must receive a globally scoped variable, so add 'global end' above (or rather use a less common global name).

This has advantages over regular 'after $time' because it does allow other events to continue processing, but that difference should be understood.

Also, when using after that takes a command in Tcl without Tk present, you have to make sure that vwait is called (or update), to kick in the event loop.


This was a horribly misguided thing... dangerously appealing and guaranteed to create snake balls.

I tried to remove the code below from this page, but some evil genius put it back!


Will suspend evaluation of current code block, but allow the event loop to continue handling pending events.

This particular version of uniqkey is my favorite because the keys generated will generally sort in time created order.

 proc uniqkey { } {
     set key   [ expr { pow(2,31) + [ clock clicks ] } ]
     set key   [ string range $key end-8 end-3 ]
     set key   [ clock seconds ]$key
     return $key
 }

 proc sleep { ms } {
     set uniq [ uniqkey ]
     set ::__sleep__tmp__$uniq 0
     after $ms set ::__sleep__tmp__$uniq 1
     vwait ::__sleep__tmp__$uniq
     unset ::__sleep__tmp__$uniq
 }

Example:

 after 4000 puts foo!
 sleep 5000
 puts bar!

-Phil Ehrens


Don't confuse this with such other sleeps as the standard Unix command-line one, or the command Expect builds in, or ...


GPS: You swilly wabbits. :)

 $ tclsh8.4
 % interp alias {} sleep {} after 
 sleep
 % sleep 400
 % interp alias {} wait {} sleep
 wait
 % wait 800

after has problems with fine-grained delays, at least on my system; its probably related to unix timeslicing.

    % time {after 1} 100
    10006 microseconds per iteration

So asking for a 1ms delay gets you something closer to 10ms. I'm in need of something much closer to accurate, so I came up with two variants of a busywaiting delay. They watch clock clicks to see when the appropriate amount of time has passed. The first, delay-bw is a pure busy-wait, while delay-ev uses the event loop.

    namespace eval delay {
        variable _i
        variable c/ms

        proc calibrate {} {
            variable c/ms
            puts "calibrating clock clicks.."
            set start [clock clicks]
            after 1000
            set end [clock clicks]
            set c/ms [expr {($end-$start)/1000.0}]
            puts "speed: [expr {${c/ms}*1000}] clicks per second"
        }

        calibrate

        # busywaiting delay
        proc delay-bw {sec} {
          variable c/ms
          set s [clock clicks]
          while {[clock clicks] < $s+(${c/ms}*$sec)} {
            # do nothing
        }

        # busywaiting "after idle" delay, using event loop 
        proc delay-ev {sec} {
          variable c/ms
          set s [clock clicks]
          set e [expr {$s+$sec*${c/ms}}]
          evwait ::delay::_i $e
          vwait ::delay::_i
          unset ::delay::_i
        }

        # worker for delay-ev
        # continually reschedules itself via "after idle" until end time
        proc evwait {var {end 0}} {
            set ct [clock clicks]
            if {$ct < $end} {
                after idle [list ::delay::evwait $var $end]
                return
            } else {
                set $var 0
            }
        }
    }

I get much more accurate delays using these:

    % time {::delay::delay-ev 1} 1000
    1283 microseconds per iteration
    % time {::delay::delay-bw 1} 1000
    1225 microseconds per iteration

JR