factor/extra/game/loop/loop.factor

118 lines
3.0 KiB
Factor
Raw Normal View History

2010-01-17 02:07:11 -05:00
! (c)2009 Joe Groff bsd license
2010-05-20 13:28:02 -04:00
USING: accessors alarms calendar continuations destructors fry
2010-05-21 13:47:42 -04:00
kernel math math.order namespaces system ui ui.gadgets.worlds ;
IN: game.loop
2009-04-28 18:23:08 -04:00
TUPLE: game-loop
2010-05-21 21:42:12 -04:00
{ tick-interval-nanos integer read-only }
tick-delegate
draw-delegate
2009-04-28 18:23:08 -04:00
{ last-tick integer }
{ running? boolean }
{ tick-number integer }
{ frame-number integer }
{ benchmark-time integer }
{ benchmark-tick-number integer }
2010-05-20 13:28:02 -04:00
{ benchmark-frame-number integer }
alarm ;
2009-04-28 18:23:08 -04:00
GENERIC: tick* ( delegate -- )
GENERIC: draw* ( tick-slice delegate -- )
SYMBOL: game-loop
2010-05-21 21:42:12 -04:00
: since-last-tick ( loop -- nanos )
last-tick>> nano-count swap - ;
2009-04-28 18:23:08 -04:00
: tick-slice ( loop -- slice )
2010-05-21 21:42:12 -04:00
[ since-last-tick ] [ tick-interval-nanos>> ] bi /f 1.0 min ;
2009-04-28 18:23:08 -04:00
CONSTANT: MAX-FRAMES-TO-SKIP 5
DEFER: stop-loop
TUPLE: game-loop-error game-loop error ;
: ?ui-error ( error -- )
ui-running? [ ui-error ] [ rethrow ] if ;
: game-loop-error ( game-loop error -- )
[ drop stop-loop ] [ \ game-loop-error boa ?ui-error ] 2bi ;
2010-05-21 21:42:12 -04:00
: fps ( fps -- nanos )
1,000,000,000 swap /i ; inline
2009-04-28 18:23:08 -04:00
<PRIVATE
: redraw ( loop -- )
[ 1 + ] change-frame-number
[ tick-slice ] [ draw-delegate>> ] bi draw* ;
2009-04-28 18:23:08 -04:00
: tick ( loop -- )
tick-delegate>> tick* ;
2009-04-28 18:23:08 -04:00
: increment-tick ( loop -- )
[ 1 + ] change-tick-number
2010-05-21 21:42:12 -04:00
dup tick-interval-nanos>> [ + ] curry change-last-tick
2009-04-28 18:23:08 -04:00
drop ;
: ?tick ( loop count -- )
2010-05-21 21:42:12 -04:00
[ nano-count >>last-tick drop ] [
over [ since-last-tick ] [ tick-interval-nanos>> ] bi >=
[ [ drop increment-tick ] [ drop tick ] [ 1 - ?tick ] 2tri ]
2009-04-28 18:23:08 -04:00
[ 2drop ] if
2009-08-11 19:15:53 -04:00
] if-zero ;
2009-04-28 18:23:08 -04:00
2010-05-21 21:42:12 -04:00
: benchmark-nanos ( loop -- nanos )
nano-count swap benchmark-time>> - ;
2009-04-28 18:23:08 -04:00
PRIVATE>
2010-05-21 21:42:12 -04:00
: reset-loop-benchmark ( loop -- loop )
nano-count >>benchmark-time
2009-04-28 18:23:08 -04:00
dup tick-number>> >>benchmark-tick-number
2010-05-21 21:42:12 -04:00
dup frame-number>> >>benchmark-frame-number ;
2009-04-28 18:23:08 -04:00
: benchmark-ticks-per-second ( loop -- n )
2010-05-21 21:42:12 -04:00
[ tick-number>> ] [ benchmark-tick-number>> - ] [ benchmark-nanos ] tri /f ;
2009-04-28 18:23:08 -04:00
: benchmark-frames-per-second ( loop -- n )
2010-05-21 21:42:12 -04:00
[ frame-number>> ] [ benchmark-frame-number>> - ] [ benchmark-nanos ] tri /f ;
2009-04-28 18:23:08 -04:00
2010-05-20 13:28:02 -04:00
: (game-tick) ( loop -- )
dup running?>>
[ [ MAX-FRAMES-TO-SKIP ?tick ] [ redraw ] bi ]
[ drop ] if ;
2010-05-21 21:42:12 -04:00
: game-tick ( loop -- )
2010-05-20 13:28:02 -04:00
dup game-loop [
[ (game-tick) ] [ game-loop-error ] recover
] with-variable ;
2009-04-28 18:23:08 -04:00
: start-loop ( loop -- )
2010-05-21 21:42:12 -04:00
nano-count >>last-tick
2009-04-28 18:23:08 -04:00
t >>running?
2010-05-21 21:42:12 -04:00
reset-loop-benchmark
[
[ '[ _ game-tick ] f ]
[ tick-interval-nanos>> nanoseconds ] bi
<alarm>
] keep [ alarm<< ] [ drop start-alarm ] 2bi ;
2009-04-28 18:23:08 -04:00
: stop-loop ( loop -- )
f >>running?
2010-05-21 21:42:12 -04:00
alarm>> stop-alarm ;
2009-04-28 18:23:08 -04:00
2010-05-21 21:42:12 -04:00
: <game-loop*> ( tick-interval-nanos tick-delegate draw-delegate -- loop )
nano-count f 0 0 nano-count 0 0 f
2009-04-28 18:23:08 -04:00
game-loop boa ;
2010-05-21 21:42:12 -04:00
: <game-loop> ( tick-interval-nanos delegate -- loop )
dup <game-loop*> ; inline
2009-04-28 18:23:08 -04:00
M: game-loop dispose
stop-loop ;
USE: vocabs.loader
{ "game.loop" "prettyprint" } "game.loop.prettyprint" require-when