2008-04-11 13:18:39 -04:00
|
|
|
! Copyright (C) 2008 Slava Pestov
|
|
|
|
! See http://factorcode.org/license.txt for BSD license.
|
2008-12-13 00:57:16 -05:00
|
|
|
USING: accessors alien alien.syntax kernel math namespaces
|
|
|
|
sequences destructors combinators threads heaps deques calendar
|
|
|
|
core-foundation core-foundation.strings
|
2008-12-13 05:57:37 -05:00
|
|
|
core-foundation.file-descriptors core-foundation.timers
|
|
|
|
core-foundation.time ;
|
2008-04-11 13:18:39 -04:00
|
|
|
IN: core-foundation.run-loop
|
|
|
|
|
|
|
|
: kCFRunLoopRunFinished 1 ; inline
|
|
|
|
: kCFRunLoopRunStopped 2 ; inline
|
|
|
|
: kCFRunLoopRunTimedOut 3 ; inline
|
|
|
|
: kCFRunLoopRunHandledSource 4 ; inline
|
|
|
|
|
|
|
|
TYPEDEF: void* CFRunLoopRef
|
2008-12-06 18:35:04 -05:00
|
|
|
TYPEDEF: void* CFRunLoopSourceRef
|
2008-04-11 13:18:39 -04:00
|
|
|
|
2008-04-11 14:20:03 -04:00
|
|
|
FUNCTION: CFRunLoopRef CFRunLoopGetMain ( ) ;
|
2008-07-14 23:21:10 -04:00
|
|
|
FUNCTION: CFRunLoopRef CFRunLoopGetCurrent ( ) ;
|
2008-04-11 14:20:03 -04:00
|
|
|
|
2008-04-11 13:18:39 -04:00
|
|
|
FUNCTION: SInt32 CFRunLoopRunInMode (
|
|
|
|
CFStringRef mode,
|
|
|
|
CFTimeInterval seconds,
|
|
|
|
Boolean returnAfterSourceHandled
|
|
|
|
) ;
|
|
|
|
|
2008-12-06 18:35:04 -05:00
|
|
|
FUNCTION: CFRunLoopSourceRef CFFileDescriptorCreateRunLoopSource (
|
|
|
|
CFAllocatorRef allocator,
|
|
|
|
CFFileDescriptorRef f,
|
|
|
|
CFIndex order
|
|
|
|
) ;
|
|
|
|
|
|
|
|
FUNCTION: void CFRunLoopAddSource (
|
|
|
|
CFRunLoopRef rl,
|
|
|
|
CFRunLoopSourceRef source,
|
|
|
|
CFStringRef mode
|
|
|
|
) ;
|
|
|
|
|
2008-12-11 23:48:19 -05:00
|
|
|
FUNCTION: void CFRunLoopRemoveSource (
|
|
|
|
CFRunLoopRef rl,
|
|
|
|
CFRunLoopSourceRef source,
|
|
|
|
CFStringRef mode
|
|
|
|
) ;
|
|
|
|
|
2008-12-12 02:11:37 -05:00
|
|
|
FUNCTION: void CFRunLoopAddTimer (
|
|
|
|
CFRunLoopRef rl,
|
|
|
|
CFRunLoopTimerRef timer,
|
|
|
|
CFStringRef mode
|
|
|
|
) ;
|
|
|
|
|
|
|
|
FUNCTION: void CFRunLoopRemoveTimer (
|
|
|
|
CFRunLoopRef rl,
|
|
|
|
CFRunLoopTimerRef timer,
|
|
|
|
CFStringRef mode
|
|
|
|
) ;
|
|
|
|
|
2008-04-11 14:20:03 -04:00
|
|
|
: CFRunLoopDefaultMode ( -- alien )
|
|
|
|
#! Ugly, but we don't have static NSStrings
|
|
|
|
\ CFRunLoopDefaultMode get-global dup expired? [
|
|
|
|
drop
|
|
|
|
"kCFRunLoopDefaultMode" <CFString>
|
|
|
|
dup \ CFRunLoopDefaultMode set-global
|
|
|
|
] when ;
|
2008-12-13 00:57:16 -05:00
|
|
|
|
|
|
|
TUPLE: run-loop fds sources timers ;
|
|
|
|
|
|
|
|
: <run-loop> ( -- run-loop )
|
|
|
|
V{ } clone V{ } clone V{ } clone \ run-loop boa ;
|
|
|
|
|
|
|
|
SYMBOL: expiry-check
|
|
|
|
|
|
|
|
: run-loop ( -- run-loop )
|
|
|
|
\ run-loop get-global not expiry-check get expired? or
|
|
|
|
[
|
|
|
|
31337 <alien> expiry-check set-global
|
|
|
|
<run-loop> dup \ run-loop set-global
|
|
|
|
] [ \ run-loop get-global ] if ;
|
|
|
|
|
|
|
|
: add-source-to-run-loop ( source -- )
|
|
|
|
[ run-loop sources>> push ]
|
|
|
|
[
|
|
|
|
CFRunLoopGetMain
|
|
|
|
swap CFRunLoopDefaultMode
|
|
|
|
CFRunLoopAddSource
|
|
|
|
] bi ;
|
|
|
|
|
|
|
|
: create-fd-source ( CFFileDescriptor -- source )
|
|
|
|
f swap 0 CFFileDescriptorCreateRunLoopSource ;
|
|
|
|
|
|
|
|
: add-fd-to-run-loop ( fd callback -- )
|
|
|
|
[
|
|
|
|
<CFFileDescriptor> |CFRelease
|
|
|
|
[ run-loop fds>> push ]
|
|
|
|
[ create-fd-source |CFRelease add-source-to-run-loop ]
|
|
|
|
bi
|
|
|
|
] with-destructors ;
|
|
|
|
|
|
|
|
: add-timer-to-run-loop ( timer -- )
|
|
|
|
[ run-loop timers>> push ]
|
|
|
|
[
|
|
|
|
CFRunLoopGetMain
|
|
|
|
swap CFRunLoopDefaultMode
|
|
|
|
CFRunLoopAddTimer
|
|
|
|
] bi ;
|
|
|
|
|
|
|
|
<PRIVATE
|
|
|
|
|
|
|
|
: ((reset-timer)) ( timer counter timestamp -- )
|
|
|
|
nip >CFAbsoluteTime CFRunLoopTimerSetNextFireDate ;
|
|
|
|
|
|
|
|
: (reset-timer) ( timer counter -- )
|
|
|
|
yield {
|
|
|
|
{ [ dup 0 = ] [ now ((reset-timer)) ] }
|
|
|
|
{ [ run-queue deque-empty? not ] [ 1- (reset-timer) ] }
|
|
|
|
{ [ sleep-queue heap-empty? ] [ 5 minutes hence ((reset-timer)) ] }
|
|
|
|
[ sleep-queue heap-peek nip micros>timestamp ((reset-timer)) ]
|
|
|
|
} cond ;
|
|
|
|
|
|
|
|
: reset-timer ( timer -- )
|
|
|
|
10 (reset-timer) ;
|
|
|
|
|
|
|
|
PRIVATE>
|
|
|
|
|
|
|
|
: reset-run-loop ( -- )
|
|
|
|
run-loop
|
|
|
|
[ timers>> [ reset-timer ] each ]
|
|
|
|
[ fds>> [ enable-all-callbacks ] each ] bi ;
|
|
|
|
|
|
|
|
: timer-callback ( -- callback )
|
|
|
|
"void" { "CFRunLoopTimerRef" "void*" } "cdecl"
|
|
|
|
[ 2drop reset-run-loop yield ] alien-callback ;
|
|
|
|
|
|
|
|
: init-thread-timer ( -- )
|
|
|
|
timer-callback <CFTimer> add-timer-to-run-loop ;
|
|
|
|
|
|
|
|
: run-one-iteration ( us -- handled? )
|
|
|
|
reset-run-loop
|
|
|
|
CFRunLoopDefaultMode
|
|
|
|
swap [ microseconds ] [ 5 minutes ] if* >CFTimeInterval
|
|
|
|
t CFRunLoopRunInMode kCFRunLoopRunHandledSource = ;
|