#lang racket/base ;; Simple lazy dataflow. (provide dataflow-graph? make-dataflow-graph dataflow-graph-edges-forward current-dataflow-subject-id dataflow-record-observation! dataflow-record-damage! dataflow-forget-subject! dataflow-repair-damage!) (require racket/set) (require "support/hash.rkt") (struct dataflow-graph (edges-forward ;; object-id -> (Setof subject-id) edges-reverse ;; subject-id -> (Setof object-id) damaged-nodes ;; Hashof object-id True ) #:mutable) (define current-dataflow-subject-id (make-parameter #f)) (define (make-dataflow-graph) (dataflow-graph (hash) (hash) (hash))) (define (dataflow-record-observation! g object-id) (define subject-id (current-dataflow-subject-id)) (when subject-id (define fwd (dataflow-graph-edges-forward g)) (set-dataflow-graph-edges-forward! g (hashset-add fwd object-id subject-id)) (define rev (dataflow-graph-edges-reverse g)) (set-dataflow-graph-edges-reverse! g (hashset-add rev subject-id object-id)))) (define (dataflow-record-damage! g object-id) (set-dataflow-graph-damaged-nodes! g (hash-set (dataflow-graph-damaged-nodes g) object-id #t))) (define (dataflow-forget-subject! g subject-id) (define rev (dataflow-graph-edges-reverse g)) (set-dataflow-graph-edges-reverse! g (hash-remove rev subject-id)) (for [(object-id (in-hashset-values rev subject-id))] (define fwd (dataflow-graph-edges-forward g)) (set-dataflow-graph-edges-forward! g (hashset-remove fwd object-id subject-id)))) (define (dataflow-repair-damage! g repair-node!) (let loop () (define workset (dataflow-graph-damaged-nodes g)) (when (not (hash-empty? workset)) (set-dataflow-graph-damaged-nodes! g (hash)) (hash-for-each workset (lambda (object-id _) (for [(subject-id (in-hashset-values (dataflow-graph-edges-forward g) object-id))] (dataflow-forget-subject! g subject-id) (parameterize ((current-dataflow-subject-id subject-id)) (repair-node! subject-id))))) (loop)))) ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; (module+ test (require rackunit) (struct cell (value) #:mutable #:methods gen:custom-write [(define (write-proc c port mode) (fprintf port "#" (cell-value c)))]) (define g #f) (define (R b) (dataflow-record-observation! g b) (cell-value b)) (define (W b v) (when (not (equal? (cell-value b) v)) (dataflow-record-damage! g b) (set-cell-value! b v))) (define (repair! b) (b)) (define-syntax-rule (V expr) (letrec ((node (cell #f)) (handler (lambda () ;; (printf "Evaluating ~a\n" 'expr) (W node expr)))) (parameterize ((current-dataflow-subject-id handler)) (handler) node))) (set! g (make-dataflow-graph)) (let* ((c (V 123)) (d (V (* (R c) 2)))) (check-equal? (list (R c) (R d)) (list 123 246)) (dataflow-repair-damage! g repair!) (check-equal? (list (R c) (R d)) (list 123 246)) (dataflow-repair-damage! g repair!) (check-equal? (list (R c) (R d)) (list 123 246)) (W c 124) (check-equal? (list (R c) (R d)) (list 124 246)) (dataflow-repair-damage! g repair!) (check-equal? (list (R c) (R d)) (list 124 248))) ;; (newline) (set! g (make-dataflow-graph)) (define xs (V (list 1 2 3 4))) (define sum (V (foldr + 0 (R xs)))) (define len (V (length (R xs)))) (define avg (V (if (zero? (R len)) (void) (/ (R sum) (R len))))) (define scale (V 1)) (define ans (V (if (zero? (R scale)) (void) (and (number? (R avg)) (/ (R avg) (R scale)))))) (define (fix! stage) ;; (printf "\n----- Stage: ~a\n" stage) (dataflow-repair-damage! g repair!) ;; (write `((xs ,(R xs)) ;; (sum ,(R sum)) ;; (len ,(R len)) ;; (avg ,(R avg)) ;; (scale ,(R scale)) ;; (ans ,(R ans)))) ;; (newline) ) (define-syntax-rule (check-results vs ...) (check-equal? (map R (list xs sum len avg scale ans)) (list vs ...))) (fix! 'initial) (check-results '(1 2 3 4) 10 4 10/4 1 10/4) (W scale 0) (fix! 'scale-zero) (check-results '(1 2 3 4) 10 4 10/4 0 (void)) (W xs (list* 9 0 (R xs))) (fix! 'with-nine-and-zero) (check-results '(9 0 1 2 3 4) 19 6 19/6 0 (void)) (W xs (list* 5 4 (cddr (R xs)))) (fix! 'with-five-and-four) (check-results '(5 4 1 2 3 4) 19 6 19/6 0 (void)) (W scale 1) (fix! 'scale-one) (check-results '(5 4 1 2 3 4) 19 6 19/6 1 19/6) (W xs '()) (fix! 'empty) (check-results '() 0 0 (void) 1 #f) (W xs (list 4 5 6)) (fix! 'four-five-six) (check-results '(4 5 6) 15 3 15/3 1 15/3))