division of labor

This commit is contained in:
Jay McCarthy 2014-11-19 10:55:00 -05:00
parent 52dc84e8bd
commit 64c9e54fdf
2 changed files with 87 additions and 108 deletions

37
chaos.rkt Normal file
View File

@ -0,0 +1,37 @@
#lang racket/base
(require racket/contract/base
data/queue
racket/generic)
(define-generics chaos
(chaos-fps chaos)
(chaos-yield chaos evt)
(chaos-inputs chaos)
(chaos-output! chaos outputs)
(chaos-label! chaos label)
(chaos-swap! chaos thunk)
#:fallbacks
[(define (chaos-fps c)
60.0)
(define (chaos-yield c e)
(sync e))
(define iq (make-queue))
(define (chaos-inputs c)
iq)
(define (chaos-output! c os)
(void))
(define (chaos-label! c l)
(void))
(define (chaos-swap! chaos thunk)
(thunk))])
(provide
gen:chaos
(contract-out
[chaos? (-> any/c boolean?)]
[chaos-fps (-> chaos? flonum?)]
[chaos-yield (-> chaos? evt? any)]
[chaos-inputs (-> chaos? queue?)]
[chaos-output! (-> chaos? vector? any)]
[chaos-label! (-> chaos? string? any)]
[chaos-swap! (-> chaos? (-> any) any)]))

158
main.rkt
View File

@ -4,7 +4,8 @@
racket/contract/base racket/contract/base
racket/flonum racket/flonum
racket/format racket/format
racket/generic) racket/generic
lux/chaos)
;; xxx abstract away sources (input events) and sinks (gui and sound)? ;; xxx abstract away sources (input events) and sinks (gui and sound)?
(define draw-mode/c (define draw-mode/c
@ -12,26 +13,11 @@
(define-generics word (define-generics word
(word-label word frame-time) (word-label word frame-time)
(word-fps word)
(word-tick word events) (word-tick word events)
(word-draw-mode word)
(word-draw! word width height dc)
(word-pause word)
(word-resume word state)
(word-stop? word)
(word-value word)
#:fallbacks #:fallbacks
[(define (word-label w frame-time) [(define (word-label w frame-time)
(lux-standard-label "Lux" frame-time)) (lux-standard-label "Lux" frame-time))
(define (word-fps w) (define (word-tick w es) (values w empty))])
60.0)
(define (word-tick w es) w)
(define (word-draw-mode w) 'draw)
(define (word-draw! w width height dc) (void))
(define (word-pause w) w)
(define (word-resume w) w)
(define (word-stop? w) #f)
(define (word-value w) w)])
(define (lux-standard-label l frame-time) (define (lux-standard-label l frame-time)
(~a l (~a l
@ -46,97 +32,53 @@
#:min-width 7 #:min-width 7
#:precision 2))) #:precision 2)))
(define current-world (make-parameter #f)) (define current-chaos (make-parameter #f))
(struct world (t ch))
(struct message (w pmz return-t return-ch time-evt)) (define (call-with-chaos c t)
(define-syntax-rule (call pmz e) (parameterize ([current-chaos c])
(call-with-continuation-barrier (t)))
(λ () (call-with-parameterization pmz (λ () e)))))
(define (start-world)
(define submit-ch (make-channel))
(define the-gui (start-gui))
(define (body old-stack)
;; xxx i dislike that i don't know if old-stack is '()
(gui-yield
the-gui
(choice-evt
(match old-stack
['()
never-evt]
[(cons old-m stack)
(match-define (message w pmz return-t return-ch time-evt) old-m)
(handle-evt
time-evt
(λ (_)
(define start-time (current-inexact-milliseconds))
(define es (gui-events the-gui))
;; I fear that (call pmz e) is slow and I do it a lot
;; here. So maybe change tick to return all this stuff?
(define new-w (call pmz (word-tick w es)))
(gui-draw!
the-gui
(call pmz (word-draw-mode new-w))
(λ (width height dc)
(call pmz (word-draw! new-w width height dc))))
(define end-time (current-inexact-milliseconds))
(define frame-time (fl- end-time start-time))
(define new-label
(call pmz (word-label new-w frame-time)))
(gui-label! the-gui new-label)
(match (call pmz (word-stop? new-w))
[#f
(define fps (call pmz (word-fps new-w)))
(define next-time (fl+ start-time (fl* (fl/ 1.0 fps) 1000.0)))
(define next-time-evt (alarm-evt next-time))
(define new-m
(message new-w pmz return-t return-ch next-time-evt))
(body (cons new-m stack))]
[#t
(thread-resume return-t)
(channel-put return-ch (word-value w))
(body
(match stack
['()
stack]
[(cons old-m stack)
(match-define (message w pmz return-t return-ch time-evt) old-m)
(cons (message (call pmz (word-resume w))
pmz return-t return-ch time-evt)
stack)]))])))])
(handle-evt
submit-ch
(λ (new-m)
(body
(cons new-m
(match old-stack
['()
old-stack]
[(cons old-m stack)
(match-define (message w pmz return-t return-ch time-evt) old-m)
(cons (message (call pmz (word-pause w))
pmz return-t return-ch time-evt)
stack)]))))))))
(define world-t
(thread
(λ ()
(body empty))))
(world world-t submit-ch))
(define (fiat-lux w) (define (fiat-lux w)
(if (current-world) (define c (current-chaos))
(fictio-fiat-lux w) (unless c
(factum-fiat-lux w))) (error 'fiat-lux "Not called within call-with-chaos"))
(factum-fiat-lux c w))
(define (factum-fiat-lux w) (define (factum-fiat-lux c w)
(parameterize ([current-world (start-world)]) (define fps (chaos-fps c))
(fictio-fiat-lux w))) (define time-incr (fl* (fl/ 1.0 fps) 1000.0))
;; xxx this isn't going to work because the server won't be listening inside tick (define (body tick-evt w)
(define (fictio-fiat-lux w) (chaos-yield
(define return-ch (make-channel)) c
(match-define (world world-t submit-ch) (current-world)) (handle-evt
(thread-resume world-t) tick-evt
(define m (message w (current-parameterization) (λ (_)
(current-thread) return-ch (define start-time (current-inexact-milliseconds))
always-evt)) (define inputs (chaos-inputs c))
(channel-put! submit-ch m) (define-values (new-w outputs) (word-tick w inputs))
(channel-get return-ch)) (match new-w
[#f
(apply values outputs)]
[_
(chaos-output! c outputs)
(define end-time (current-inexact-milliseconds))
(define frame-time (fl- end-time start-time))
(define new-label
(word-label new-w frame-time))
(chaos-label! c new-label)
(define next-time (fl+ start-time time-incr))
(define next-tick-evt (alarm-evt next-time))
(body next-tick-evt new-w)])))))
(chaos-swap! c (λ () (body always-evt w))))
(provide gen:word
(contract-out
[lux-standard-label
(-> string? flonum?
string?)]
[call-with-chaos
(-> chaos? (-> any)
any)]
[fiat-lux
(-> word?
any)]))