diff --git a/qi-doc/scribblings/list-operations.scrbl b/qi-doc/scribblings/list-operations.scrbl new file mode 100644 index 000000000..2b70c1181 --- /dev/null +++ b/qi-doc/scribblings/list-operations.scrbl @@ -0,0 +1,118 @@ +#lang scribble/doc +@require[scribble/manual + (for-label racket/list + racket/base)] + +@title{List Operations} + +@defmodule[qi/list] + +This module defines bindings that can leverage stream fusion / +deforestation optimization when found in succession within a +flow. When not part of optimized flow, their behavior is identical to +the bindings of the same name from @racketmodname[racket/base] and +@racketmodname[racket/list]. + +The bindings are categorized based on their intended usage inside the +deforested pipeline. + +@section{Producers} + +@defproc*[(((range (end real?)) list?) + ((range (start real?) (end real?) (step real? 1)) list?))]{ + +Deforestable version of @racket[range] from @racketmodname[racket/list]. + +} + +@section{Transformers} + +@defproc[(filter (pred procedure?) (lst list?)) list?]{ + +Deforestable version of @racket[filter] from @racketmodname[racket/base]. + +} + +@defproc[(map (proc procedure?) (lst list?) ...+) list?]{ + +Deforestable version of @racket[map] from @racketmodname[racket/base]. + +} + +@defproc[(filter-map (proc procedure?) (lst list?) ...+) list?]{ + +Deforestable version of @racket[filter-map] from @racketmodname[racket/list]. + +} + +@defproc*[(((take (lst list?) (pos exact-nonnegative-integer?)) list?) + ((take (lst any/c) (pos exact-nonnegative-integer?)) list?))]{ + +Deforestable version of @racket[take] from @racketmodname[racket/list]. + +} + +@section{Consumers} + +@defproc[(foldl (proc procedure?) (init any/c) (lst list?) ...+) any/c]{ + +Deforestable version of @racket[foldl] from @racketmodname[racket/base]. + +} + +@defproc[(foldr (proc procedure?) (init any/c) (lst list?) ...+) any/c]{ + +Deforestable version of @racket[foldr] from @racketmodname[racket/base]. + +} + +@defproc[(car (p pair?)) any/c]{ + +Deforestable version of @racket[car] from @racketmodname[racket/base]. + +} + +@defproc[(cadr (v (cons/c any/c pair?))) any/c]{ + +Deforestable version of @racket[cadr] from @racketmodname[racket/base]. + +} + +@defproc[(caddr (v (cons/c any/c (cons/c any/c pair?)))) any/c]{ + +Deforestable version of @racket[caddr] from @racketmodname[racket/base]. + +} + +@defproc[(cadddr (v (cons/c any/c (cons/c any/c (cons/c any/c pair?))))) any/c]{ + +Deforestable version of @racket[cadddr] from @racketmodname[racket/base]. + +} + +@defproc*[(((list-ref (lst list?) (pos exact-nonnegative-integer?)) any/c) + ((list-ref (lst pair?) (pos exact-nonnegative-integer?)) any/c))]{ + +Deforestable version of @racket[list-ref] from @racketmodname[racket/base]. + +} + +@defproc[(length (lst list?)) exact-nonnegative-integer?]{ + +Deforestable version of @racket[length] from @racketmodname[racket/base]. + +} + +@defproc[(empty? (v any/c)) boolean?]{ + +Deforestable version of @racket[empty?] from @racketmodname[racket/list]. + +} + +@defproc[(null? (v any/c)) boolean?]{ + +Deforestable version of @racket[null?] from @racketmodname[racket/base]. + +} + + diff --git a/qi-doc/scribblings/qi.scrbl b/qi-doc/scribblings/qi.scrbl index dc7384f32..c76c7f85c 100644 --- a/qi-doc/scribblings/qi.scrbl +++ b/qi-doc/scribblings/qi.scrbl @@ -40,6 +40,7 @@ This site hosts @emph{user} documentation. If you are interested in contributing @include-section["tutorial.scrbl"] @include-section["interface.scrbl"] @include-section["forms.scrbl"] +@include-section["list-operations.scrbl"] @include-section["macros.scrbl"] @include-section["field-guide.scrbl"] @include-section["principles.scrbl"] diff --git a/qi-lib/flow/core/compiler.rkt b/qi-lib/flow/core/compiler.rkt index 85edd46c9..c8101307b 100644 --- a/qi-lib/flow/core/compiler.rkt +++ b/qi-lib/flow/core/compiler.rkt @@ -3,497 +3,16 @@ (provide (for-syntax compile-flow normalize-pass)) (require (for-syntax racket/base - syntax/parse - racket/match - (only-in racket/list make-list) - "syntax.rkt" - "../aux-syntax.rkt" - "strategy.rkt" - "private/form-property.rkt") - "impl.rkt" - "passes.rkt" - "normalize.rkt" - (only-in racket/list make-list) - racket/function - racket/undefined - (prefix-in fancy: fancy-app) - racket/list) + syntax/parse) + "compiler/1000-qi0.rkt" + "compiler/2000-bindings.rkt" + "compiler/0010-normalize.rkt" + "compiler/0100-deforest.rkt" + "passes.rkt") (begin-for-syntax ;; note: this does not return compiled code but instead, ;; syntax whose expansion compiles the code (define (compile-flow stx) - (run-passes stx)) - - (define-and-register-pass 1000 (qi0-wrapper stx) - (syntax-parse stx - (ex #'(qi0->racket ex)))) - - ) - -;; Transformation rules for the `as` binding form: -;; -;; 1. escape to wrap outermost ~> with let and re-enter -;; -;; (~> flo ... (... (as name) ...)) -;; ... -;; ↓ -;; ... -;; (esc (let ([name (void)]) -;; (☯ original-flow))) -;; -;; 2. as → set! -;; -;; (as name) -;; ... -;; ↓ -;; ... -;; (~> (esc (λ (x) (set! name x))) ⏚) -;; -;; 3. Overall transformation: -;; -;; (~> flo ... (... (as name) ...)) -;; ... -;; ↓ -;; ... -;; (esc (let ([name (void)]) -;; (☯ (~> flo ... (... (~> (esc (λ (x) (set! name x))) ⏚) ...))))) - -(begin-for-syntax - - ;; (as name) → (~> (esc (λ (x) (set! name x))) ⏚) - ;; TODO: use a box instead of set! - (define (rewrite-all-bindings stx) - (find-and-map/qi (syntax-parser - [((~datum as) x ...) - #:with (x-val ...) (generate-temporaries (attribute x)) - #'(thread (esc (λ (x-val ...) (set! x x-val) ...)) ground)] - [_ this-syntax]) - stx)) - - (define (bound-identifiers stx) - (let ([ids null]) - (find-and-map/qi (syntax-parser - [((~datum as) x ...) - (begin - (set! ids (append (attribute x) ids)) - ;; we don't need to traverse further - #f)] - [_ this-syntax]) - stx) - ids)) - - ;; wrap stx with (let ([v undefined] ...) stx) for v ∈ ids - (define (wrap-with-scopes stx ids) - (with-syntax ([(v ...) ids]) - #`(let ([v undefined] ...) #,stx))) - - (define-and-register-pass 2000 (bindings stx) - ;; TODO: use syntax-parse and match ~> specifically. - ;; Since macros are expanded "outside in," presumably - ;; it will naturally wrap the outermost ~> - (wrap-with-scopes (rewrite-all-bindings stx) - (bound-identifiers stx)))) - -(define-syntax (qi0->racket stx) - ;; this is a macro so it receives the entire expression - ;; (qi0->racket ...). We use cadr here to parse the - ;; contained expression. - (syntax-parse (cadr (syntax->list stx)) - - ;;;;;;;;;;;;;;;;;;;;;;;;;;;;; - ;;;; Core language forms ;;;; - ;;;;;;;;;;;;;;;;;;;;;;;;;;;;; - - [((~datum gen) ex:expr ...) - #'(λ _ (values ex ...))] - ;; pass-through (identity flow) - [(~datum _) #'values] - ;; routing - [(~or* (~datum ⏚) (~datum ground)) ; NOTE: technically not core - #'(qi0->racket (select))] - [((~or* (~datum ~>) (~datum thread)) onex:clause ...) - #`(compose . #,(reverse - (syntax->list - #'((qi0->racket onex) ...))))] - [e:relay-form (relay-parser #'e)] - [e:tee-form (tee-parser #'e)] - ;; map and filter - [e:amp-form (amp-parser #'e)] ; NOTE: technically not core - [e:pass-form (pass-parser #'e)] ; NOTE: technically not core - ;; prisms - [e:sep-form (sep-parser #'e)] - [(~or* (~datum ▽) (~datum collect)) - #'list] - ;; predicates - [(~or* (~datum NOT) (~datum !)) - #'not] - [(~datum XOR) - #'parity-xor] - [((~datum and) onex:clause ...) - #'(conjoin (qi0->racket onex) ...)] - [((~datum or) onex:clause ...) - #'(disjoin (qi0->racket onex) ...)] - [((~datum not) onex:clause) ; NOTE: technically not core - #'(negate (qi0->racket onex))] - [((~datum all) onex:clause) - #`(give (curry andmap (qi0->racket onex)))] - [((~datum any) onex:clause) - #'(give (curry ormap (qi0->racket onex)))] - - ;; selection - [e:select-form (select-parser #'e)] - [e:block-form (block-parser #'e)] - [e:group-form (group-parser #'e)] - ;; conditionals - [e:if-form (if-parser #'e)] - [e:sieve-form (sieve-parser #'e)] - [e:partition-form (partition-parser #'e)] - ;; exceptions - [e:try-form (try-parser #'e)] - ;; folds - [e:fold-left-form (fold-left-parser #'e)] - [e:fold-right-form (fold-right-parser #'e)] - ;; high-level routing - [e:fanout-form (fanout-parser #'e)] - ;; looping - [e:feedback-form (feedback-parser #'e)] - [e:loop-form (loop-parser #'e)] - [((~datum loop2) pred:clause mapex:clause combex:clause) - #'(letrec ([loop2 (qi0->racket (if pred - (~> (== (-< (esc cdr) - (~> (esc car) mapex)) _) - (group 1 _ combex) - (esc loop2)) - (select 2)))]) - loop2)] - ;; towards universality - [(~datum appleye) - #'call] - [e:clos-form (clos-parser #'e)] - ;; escape hatch for racket expressions or anything - ;; to be "passed through" - [((~datum esc) ex:expr) - #'ex] - - ;;; Miscellaneous - - ;; Partial application with syntactically pre-supplied arguments - ;; in a blanket template - ;; Note: at this point it's already been parsed/validated - ;; by the expander and we don't need to worry about checking - ;; the syntax at the compiler level - [((~datum #%blanket-template) e) - (blanket-template-form-parser this-syntax)] - - ;; Fine-grained template-based application - ;; This handles templates that indicate a specific number of template - ;; variables (i.e. expected arguments). The semantics of template-based - ;; application here is fulfilled by the fancy-app module. In order to use - ;; it, we simply use the #%app macro provided by fancy-app instead of the - ;; implicit one used for function application in racket/base. - ;; "prarg" = "pre-supplied argument" - [((~datum #%fine-template) (prarg-pre ... (~datum _) prarg-post ...)) - #'(fancy:#%app prarg-pre ... _ prarg-post ...)] - - ;; If in the course of optimization we ever end up with a fully - ;; simplified host expression, the compiler would a priori reject it as - ;; not being a core Qi expression. So we add this extra rule here - ;; to simply pass this expression through. - ;; TODO: should `#%host-expression` be formally declared as being part - ;; of the core language by including it in the syntax-spec grammar - ;; in extended/expander.rkt? - [((~datum #%host-expression) hex) - this-syntax])) - -;; The form-specific parsers, which are delegated to from -;; the qi0->racket macro: - -#| -A note on error handling: - -Some forms, in addition to handling legitimate syntax, also have -catch-all versions that exist purely to provide a helpful message -indicating a syntax error. We do this since a priori the qi0->racket macro -would ignore syntax that doesn't match any pattern. Yet, for all of -these named forms, we know that (or at least, it is prudent to assume -that) the user intended to employ that particular form of the DSL. So -instead of allowing it to fall through for interpretation as Racket -code, which would yield potentially inscrutable errors, the catch-all -forms allow us to provide appropriate error messages at the level of -the DSL. - -|# - -(begin-for-syntax - - (define (sep-parser stx) - (syntax-parse stx - [_:id - #'(qi0->racket (if (esc list?) - (#%fine-template (apply values _)) - (#%fine-template (raise-argument-error '△ - "list?" - _))))] - [(_ onex:clause) - #'(λ (v . vs) - ((qi0->racket (~> △ (>< (#%fine-template (apply (qi0->racket onex) _ vs))))) v))])) - - (define (select-parser stx) - (syntax-parse stx - [(_ n:number ...) #'(qi0->racket (-< (esc (arg n)) ...))])) - - (define (block-parser stx) - (syntax-parse stx - [(_ n:number ...) - #'(qi0->racket (~> (esc (except-args n ...)) - △))])) - - (define (group-parser stx) - (syntax-parse stx - [(_ n:expr - selection-onex:clause - remainder-onex:clause) - #'(loom-compose (qi0->racket selection-onex) - (qi0->racket remainder-onex) - n)] - [_:id - #'(λ (n selection-flo remainder-flo . vs) - (apply (qi0->racket (group n - (esc selection-flo) - (esc remainder-flo))) vs))])) - - (define (sieve-parser stx) - (syntax-parse stx - [(_ condition:clause - sonex:clause - ronex:clause) - #'(qi0->racket (-< (~> (pass condition) sonex) - (~> (pass (not condition)) ronex)))] - [_:id - ;; sieve can be a core form once bindings - ;; are introduced into the language - #'(λ (condition sonex ronex . args) - (apply (qi0->racket (-< (~> (pass (esc condition)) (esc sonex)) - (~> (pass (not (esc condition))) (esc ronex)))) - args))])) - - (define (partition-parser stx) - (syntax-parse stx - [(_:id) - #'(qi0->racket ground)] - [(_ [cond:clause body:clause]) - #'(qi0->racket (~> (pass cond) body))] - [(_ [cond:clause body:clause] ...+) - #:with c+bs #'(list (cons (qi0->racket cond) (qi0->racket body)) ...) - #'(qi0->racket (#%blanket-template (partition-values c+bs __)))])) - - (define (try-parser stx) - (syntax-parse stx - [(_ flo - [error-condition-flo error-handler-flo] - ...+) - #'(λ args - (with-handlers ([(qi0->racket error-condition-flo) - (λ (e) - ;; TODO: may be good to support reference to the - ;; error via a binding / syntax parameter - (apply (qi0->racket error-handler-flo) args))] - ...) - (apply (qi0->racket flo) args)))])) - - (define (if-parser stx) - (syntax-parse stx - [(_ consequent:clause - alternative:clause) - #'(λ (f . args) - (if (apply f args) - (apply (qi0->racket consequent) args) - (apply (qi0->racket alternative) args)))] - [(_ condition:clause - consequent:clause - alternative:clause) - #'(λ args - (if (apply (qi0->racket condition) args) - (apply (qi0->racket consequent) args) - (apply (qi0->racket alternative) args)))])) - - (define (fanout-parser stx) - (syntax-parse stx - [_:id #'repeat-values] - [(_ n:number) - ;; a slightly more efficient compile-time implementation - ;; for literally indicated N - ;; TODO: implement this as an optimization instead - #`(λ args - (apply values - (append #,@(make-list (syntax->datum #'n) #'args))) )] - [(_ n:expr) - #'(lambda args - (apply values - (apply append - (make-list n args))))])) - - (define (feedback-parser stx) - (syntax-parse stx - [(_ ((~datum while) tilex:clause) - ((~datum then) thenex:clause) - onex:clause) - #'(feedback-while (qi0->racket onex) - (qi0->racket tilex) - (qi0->racket thenex))] - [(_ ((~datum while) tilex:clause) - ((~datum then) thenex:clause)) - #'(λ (f . args) - (apply (qi0->racket (feedback (while tilex) (then thenex) (esc f))) - args))] - [(_ ((~datum while) tilex:clause) onex:clause) - #'(qi0->racket (feedback (while tilex) (then _) onex))] - [(_ ((~datum while) tilex:clause)) - #'(qi0->racket (feedback (while tilex) (then _)))] - [(_ n:expr - ((~datum then) thenex:clause) - onex:clause) - #'(lambda args - (apply (feedback-times (qi0->racket onex) n (qi0->racket thenex)) - args))] - [(_ n:expr - ((~datum then) thenex:clause)) - #'(λ (f . args) - (apply (qi0->racket (feedback n (then thenex) (esc f))) args))] - [(_ n:expr onex:clause) - #'(qi0->racket (feedback n (then _) onex))] - [(_ onex:clause) - #'(λ (n . args) - (apply (qi0->racket (feedback n onex)) args))] - [_:id - #'(λ (n flo . args) - (apply (qi0->racket (feedback n (esc flo))) - args))])) - - (define (tee-parser stx) - (syntax-parse stx - [((~or* (~datum -<) (~datum tee)) onex:clause ...) - #'(λ args - (apply values - (append (values->list - (apply (qi0->racket onex) args)) - ...)))] - [(~or* (~datum -<) (~datum tee)) - #'repeat-values])) - - (define (relay-parser stx) - (syntax-parse stx - [((~or* (~datum ==) (~datum relay)) onex:clause ...) - #'(relay (qi0->racket onex) ...)] - [(~or* (~datum ==) (~datum relay)) - ;; review this – this "map" behavior may not be natural - ;; for relay. And map-values should probably end up being - ;; used in a compiler optimization - #'map-values])) - - (define (amp-parser stx) - (syntax-parse stx - [_:id - #'(qi0->racket ==)] - [(_ onex:clause) - #'(curry map-values (qi0->racket onex))])) - - (define (pass-parser stx) - (syntax-parse stx - [_:id - #'filter-values] - [(_ onex:clause) - #'(curry filter-values (qi0->racket onex))])) - - (define (fold-left-parser stx) - (syntax-parse stx - [_:id - #'foldl-values] - [(_ fn init) - #'(qi0->racket (~> (-< (gen (qi0->racket fn) - (qi0->racket init)) - _) - >>))] - [(_ fn) - #'(qi0->racket (>> fn (gen ((qi0->racket fn)))))])) - - (define (fold-right-parser stx) - (syntax-parse stx - [_:id - #'foldr-values] - [(_ fn init) - #'(qi0->racket (~> (-< (gen (qi0->racket fn) - (qi0->racket init)) - _) - <<))] - [(_ fn) - #'(qi0->racket (<< fn (gen ((qi0->racket fn)))))])) - - (define (loop-parser stx) - (syntax-parse stx - [(_ pred:clause mapex:clause combex:clause retex:clause) - #'(letrec ([loop (qi0->racket (if pred - (~> (group 1 mapex (esc loop)) - combex) - retex))]) - loop)] - [(_ pred:clause mapex:clause combex:clause) - #'(qi0->racket (loop pred mapex combex ⏚))] - [(_ pred:clause mapex:clause) - #'(qi0->racket (loop pred mapex _ ⏚))] - [(_ mapex:clause) - #'(qi0->racket (loop (gen #t) mapex _ ⏚))] - [_:id #'(λ (predf mapf combf retf . args) - (apply (qi0->racket (loop (esc predf) - (esc mapf) - (esc combf) - (esc retf))) - args))])) - - (define (clos-parser stx) - (syntax-parse stx - [_:id - #:do [(define chirality (syntax-property stx 'chirality))] - (if (and chirality (eq? chirality 'right)) - #'(λ (f . args) (apply curryr f args)) - #'(λ (f . args) (apply curry f args)))] - [(_ onex:clause) - #:do [(define chirality (syntax-property stx 'chirality))] - (if (and chirality (eq? chirality 'right)) - #'(λ args - (qi0->racket (~> (-< _ (~> (gen args) △)) - onex))) - #'(λ args - (qi0->racket (~> (-< (~> (gen args) △) _) - onex))))])) - - (define (blanket-template-form-parser stx) - (syntax-parse stx - ;; "prarg" = "pre-supplied argument" - ;; Note: use of currying here doesn't play well with bindings - ;; because curry / curryr immediately evaluate their arguments - ;; and resolve any references to bindings at compile time. - ;; That's why we use a lambda which delays evaluation until runtime - ;; when the reference is actually resolvable. See "anaphoric references" - ;; in the compiler meeting notes, - ;; "The Artist Formerly Known as Bindingspec" - [((~datum #%blanket-template) - (natex prarg-pre ...+ (~datum __) prarg-post ...+)) - ;; "(curry (curryr ...) ...)" - #'(lambda largs - (apply - (lambda rargs - ((kw-helper natex rargs) prarg-post ...)) - prarg-pre ... - largs))] - [((~datum #%blanket-template) (natex prarg-pre ...+ (~datum __))) - ;; "curry" - #'(lambda args - (apply natex prarg-pre ... args))] - [((~datum #%blanket-template) - (natex (~datum __) prarg-post ...+)) - ;; "curryr" - #'(lambda args - ((kw-helper natex args) prarg-post ...))]))) + (run-passes stx))) diff --git a/qi-lib/flow/core/normalize.rkt b/qi-lib/flow/core/compiler/0010-normalize.rkt similarity index 95% rename from qi-lib/flow/core/normalize.rkt rename to qi-lib/flow/core/compiler/0010-normalize.rkt index 387aac0a1..fce732c04 100644 --- a/qi-lib/flow/core/normalize.rkt +++ b/qi-lib/flow/core/compiler/0010-normalize.rkt @@ -4,9 +4,9 @@ (require (for-syntax racket/base syntax/parse - "strategy.rkt" - "private/form-property.rkt") - "passes.rkt") + "../strategy.rkt" + "../private/form-property.rkt") + "../passes.rkt") ;; 0. "Qi-normal form" (begin-for-syntax diff --git a/qi-lib/flow/core/compiler/0100-deforest.rkt b/qi-lib/flow/core/compiler/0100-deforest.rkt new file mode 100644 index 000000000..6ab99823e --- /dev/null +++ b/qi-lib/flow/core/compiler/0100-deforest.rkt @@ -0,0 +1,5 @@ +#lang racket/base + +(provide (for-syntax deforest-pass)) + +(require "deforest/cps.rkt") diff --git a/qi-lib/flow/core/compiler/1000-qi0.rkt b/qi-lib/flow/core/compiler/1000-qi0.rkt new file mode 100644 index 000000000..ee2e93e36 --- /dev/null +++ b/qi-lib/flow/core/compiler/1000-qi0.rkt @@ -0,0 +1,460 @@ +#lang racket/base + +(require "../passes.rkt" + (prefix-in fancy: fancy-app) + "../impl.rkt" + racket/function + racket/list + (for-syntax racket/base + syntax/parse + "../syntax.rkt" + "../../aux-syntax.rkt" + (only-in racket/list make-list) + )) + +(begin-for-syntax + + (define-and-register-pass 1000 (qi0-wrapper stx) + (syntax-parse stx + (ex #'(qi0->racket ex)))) + + ) + +(define-syntax (qi0->racket stx) + ;; this is a macro so it receives the entire expression + ;; (qi0->racket ...). We use cadr here to parse the + ;; contained expression. + (syntax-parse (cadr (syntax->list stx)) + + ;;;;;;;;;;;;;;;;;;;;;;;;;;;;; + ;;;; Core language forms ;;;; + ;;;;;;;;;;;;;;;;;;;;;;;;;;;;; + + [((~datum gen) ex:expr ...) + #'(λ _ (values ex ...))] + ;; pass-through (identity flow) + [(~datum _) #'values] + ;; routing + [(~or* (~datum ⏚) (~datum ground)) ; NOTE: technically not core + #'(qi0->racket (select))] + [((~or* (~datum ~>) (~datum thread)) onex:clause ...) + #`(compose . #,(reverse + (syntax->list + #'((qi0->racket onex) ...))))] + [e:relay-form (relay-parser #'e)] + [e:tee-form (tee-parser #'e)] + ;; map and filter + [e:amp-form (amp-parser #'e)] ; NOTE: technically not core + [e:pass-form (pass-parser #'e)] ; NOTE: technically not core + ;; prisms + [e:sep-form (sep-parser #'e)] + [(~or* (~datum ▽) (~datum collect)) + #'list] + ;; predicates + [(~or* (~datum NOT) (~datum !)) + #'not] + [(~datum XOR) + #'parity-xor] + [((~datum and) onex:clause ...) + #'(conjoin (qi0->racket onex) ...)] + [((~datum or) onex:clause ...) + #'(disjoin (qi0->racket onex) ...)] + [((~datum not) onex:clause) ; NOTE: technically not core + #'(negate (qi0->racket onex))] + [((~datum all) onex:clause) + #`(give (curry andmap (qi0->racket onex)))] + [((~datum any) onex:clause) + #'(give (curry ormap (qi0->racket onex)))] + + ;; selection + [e:select-form (select-parser #'e)] + [e:block-form (block-parser #'e)] + [e:group-form (group-parser #'e)] + ;; conditionals + [e:if-form (if-parser #'e)] + [e:sieve-form (sieve-parser #'e)] + [e:partition-form (partition-parser #'e)] + ;; exceptions + [e:try-form (try-parser #'e)] + ;; folds + [e:fold-left-form (fold-left-parser #'e)] + [e:fold-right-form (fold-right-parser #'e)] + ;; high-level routing + [e:fanout-form (fanout-parser #'e)] + ;; looping + [e:feedback-form (feedback-parser #'e)] + [e:loop-form (loop-parser #'e)] + [((~datum loop2) pred:clause mapex:clause combex:clause) + #'(letrec ([loop2 (qi0->racket (if pred + (~> (== (-< (esc cdr) + (~> (esc car) mapex)) _) + (group 1 _ combex) + (esc loop2)) + (select 2)))]) + loop2)] + ;; towards universality + [(~datum appleye) + #'call] + [e:clos-form (clos-parser #'e)] + [e:deforestable-form (deforestable-parser #'e)] + ;; escape hatch for racket expressions or anything + ;; to be "passed through" + [((~datum esc) ex:expr) + #'ex] + + ;;; Miscellaneous + + ;; Partial application with syntactically pre-supplied arguments + ;; in a blanket template + [((~datum #%blanket-template) e) + (blanket-template-form-parser this-syntax)] + + ;; Fine-grained template-based application + ;; This handles templates that indicate a specific number of template + ;; variables (i.e. expected arguments). The semantics of template-based + ;; application here is fulfilled by the fancy-app module. In order to use + ;; it, we simply use the #%app macro provided by fancy-app instead of the + ;; implicit one used for function application in racket/base. + ;; "prarg" = "pre-supplied argument" + [((~datum #%fine-template) (prarg-pre ... (~datum _) prarg-post ...)) + #'(fancy:#%app prarg-pre ... _ prarg-post ...)] + + ;; If in the course of optimization we ever end up with a fully + ;; simplified host expression, the compiler would a priori reject it as + ;; not being a core Qi expression. So we add this extra rule here + ;; to simply pass this expression through. + ;; TODO: should `#%host-expression` be formally declared as being part + ;; of the core language by including it in the syntax-spec grammar + ;; in extended/expander.rkt? + [((~datum #%host-expression) hex) + this-syntax])) + +;; The form-specific parsers, which are delegated to from +;; the qi0->racket macro: + +#| +A note on error handling: + +Some forms, in addition to handling legitimate syntax, also have +catch-all versions that exist purely to provide a helpful message +indicating a syntax error. We do this since a priori the qi0->racket macro +would ignore syntax that doesn't match any pattern. Yet, for all of +these named forms, we know that (or at least, it is prudent to assume +that) the user intended to employ that particular form of the DSL. So +instead of allowing it to fall through for interpretation as Racket +code, which would yield potentially inscrutable errors, the catch-all +forms allow us to provide appropriate error messages at the level of +the DSL. + +|# + +(begin-for-syntax + + (define (sep-parser stx) + (syntax-parse stx + [_:id + #'(qi0->racket (if (esc list?) + (#%fine-template (apply values _)) + (#%fine-template (raise-argument-error '△ + "list?" + _))))] + [(_ onex:clause) + #'(λ (v . vs) + ((qi0->racket (~> △ (>< (#%fine-template (apply (qi0->racket onex) _ vs))))) v))])) + + (define (select-parser stx) + (syntax-parse stx + [(_ n:number ...) #'(qi0->racket (-< (esc (arg n)) ...))])) + + (define (block-parser stx) + (syntax-parse stx + [(_ n:number ...) + #'(qi0->racket (~> (esc (except-args n ...)) + △))])) + + (define (group-parser stx) + (syntax-parse stx + [(_ n:expr + selection-onex:clause + remainder-onex:clause) + #'(loom-compose (qi0->racket selection-onex) + (qi0->racket remainder-onex) + n)] + [_:id + #'(λ (n selection-flo remainder-flo . vs) + (apply (qi0->racket (group n + (esc selection-flo) + (esc remainder-flo))) vs))])) + + (define (sieve-parser stx) + (syntax-parse stx + [(_ condition:clause + sonex:clause + ronex:clause) + #'(qi0->racket (-< (~> (pass condition) sonex) + (~> (pass (not condition)) ronex)))] + [_:id + ;; sieve can be a core form once bindings + ;; are introduced into the language + #'(λ (condition sonex ronex . args) + (apply (qi0->racket (-< (~> (pass (esc condition)) (esc sonex)) + (~> (pass (not (esc condition))) (esc ronex)))) + args))])) + + (define (partition-parser stx) + (syntax-parse stx + [(_:id) + #'(qi0->racket ground)] + [(_ [cond:clause body:clause]) + #'(qi0->racket (~> (pass cond) body))] + [(_ [cond:clause body:clause] ...+) + #:with c+bs #'(list (cons (qi0->racket cond) (qi0->racket body)) ...) + #'(qi0->racket (#%blanket-template (partition-values c+bs __)))])) + + (define (try-parser stx) + (syntax-parse stx + [(_ flo + [error-condition-flo error-handler-flo] + ...+) + #'(λ args + (with-handlers ([(qi0->racket error-condition-flo) + (λ (e) + ;; TODO: may be good to support reference to the + ;; error via a binding / syntax parameter + (apply (qi0->racket error-handler-flo) args))] + ...) + (apply (qi0->racket flo) args)))])) + + (define (if-parser stx) + (syntax-parse stx + [(_ consequent:clause + alternative:clause) + #'(λ (f . args) + (if (apply f args) + (apply (qi0->racket consequent) args) + (apply (qi0->racket alternative) args)))] + [(_ condition:clause + consequent:clause + alternative:clause) + #'(λ args + (if (apply (qi0->racket condition) args) + (apply (qi0->racket consequent) args) + (apply (qi0->racket alternative) args)))])) + + (define (fanout-parser stx) + (syntax-parse stx + [_:id #'repeat-values] + [(_ n:number) + ;; a slightly more efficient compile-time implementation + ;; for literally indicated N + ;; TODO: implement this as an optimization instead + #`(λ args + (apply values + (append #,@(make-list (syntax->datum #'n) #'args))) )] + [(_ n:expr) + #'(lambda args + (apply values + (apply append + (make-list n args))))])) + + (define (feedback-parser stx) + (syntax-parse stx + [(_ ((~datum while) tilex:clause) + ((~datum then) thenex:clause) + onex:clause) + #'(feedback-while (qi0->racket onex) + (qi0->racket tilex) + (qi0->racket thenex))] + [(_ ((~datum while) tilex:clause) + ((~datum then) thenex:clause)) + #'(λ (f . args) + (apply (qi0->racket (feedback (while tilex) (then thenex) (esc f))) + args))] + [(_ ((~datum while) tilex:clause) onex:clause) + #'(qi0->racket (feedback (while tilex) (then _) onex))] + [(_ ((~datum while) tilex:clause)) + #'(qi0->racket (feedback (while tilex) (then _)))] + [(_ n:expr + ((~datum then) thenex:clause) + onex:clause) + #'(lambda args + (apply (feedback-times (qi0->racket onex) n (qi0->racket thenex)) + args))] + [(_ n:expr + ((~datum then) thenex:clause)) + #'(λ (f . args) + (apply (qi0->racket (feedback n (then thenex) (esc f))) args))] + [(_ n:expr onex:clause) + #'(qi0->racket (feedback n (then _) onex))] + [(_ onex:clause) + #'(λ (n . args) + (apply (qi0->racket (feedback n onex)) args))] + [_:id + #'(λ (n flo . args) + (apply (qi0->racket (feedback n (esc flo))) + args))])) + + (define (tee-parser stx) + (syntax-parse stx + [((~or* (~datum -<) (~datum tee)) onex:clause ...) + #'(λ args + (apply values + (append (values->list + (apply (qi0->racket onex) args)) + ...)))] + [(~or* (~datum -<) (~datum tee)) + #'repeat-values])) + + (define (relay-parser stx) + (syntax-parse stx + [((~or* (~datum ==) (~datum relay)) onex:clause ...) + #'(relay (qi0->racket onex) ...)] + [(~or* (~datum ==) (~datum relay)) + ;; review this – this "map" behavior may not be natural + ;; for relay. And map-values should probably end up being + ;; used in a compiler optimization + #'map-values])) + + (define (amp-parser stx) + (syntax-parse stx + [_:id + #'(qi0->racket ==)] + [(_ onex:clause) + #'(curry map-values (qi0->racket onex))])) + + (define (pass-parser stx) + (syntax-parse stx + [_:id + #'filter-values] + [(_ onex:clause) + #'(curry filter-values (qi0->racket onex))])) + + (define (fold-left-parser stx) + (syntax-parse stx + [_:id + #'foldl-values] + [(_ fn init) + #'(qi0->racket (~> (-< (gen (qi0->racket fn) + (qi0->racket init)) + _) + >>))] + [(_ fn) + #'(qi0->racket (>> fn (gen ((qi0->racket fn)))))])) + + (define (fold-right-parser stx) + (syntax-parse stx + [_:id + #'foldr-values] + [(_ fn init) + #'(qi0->racket (~> (-< (gen (qi0->racket fn) + (qi0->racket init)) + _) + <<))] + [(_ fn) + #'(qi0->racket (<< fn (gen ((qi0->racket fn)))))])) + + (define (loop-parser stx) + (syntax-parse stx + [(_ pred:clause mapex:clause combex:clause retex:clause) + #'(letrec ([loop (qi0->racket (if pred + (~> (group 1 mapex (esc loop)) + combex) + retex))]) + loop)] + [(_ pred:clause mapex:clause combex:clause) + #'(qi0->racket (loop pred mapex combex ⏚))] + [(_ pred:clause mapex:clause) + #'(qi0->racket (loop pred mapex _ ⏚))] + [(_ mapex:clause) + #'(qi0->racket (loop (gen #t) mapex _ ⏚))] + [_:id #'(λ (predf mapf combf retf . args) + (apply (qi0->racket (loop (esc predf) + (esc mapf) + (esc combf) + (esc retf))) + args))])) + + (define (clos-parser stx) + (syntax-parse stx + [_:id + #:do [(define chirality (syntax-property stx 'chirality))] + (if (and chirality (eq? chirality 'right)) + #'(λ (f . args) (apply curryr f args)) + #'(λ (f . args) (apply curry f args)))] + [(_ onex:clause) + #:do [(define chirality (syntax-property stx 'chirality))] + (if (and chirality (eq? chirality 'right)) + #'(λ args + (qi0->racket (~> (-< _ (~> (gen args) △)) + onex))) + #'(λ args + (qi0->racket (~> (-< (~> (gen args) △) _) + onex))))])) + + (define (deforestable-parser stx) + (syntax-parse stx + [((~datum #%deforestable) (~datum filter) (proc:clause)) + #'(lambda (v) + (filter (qi0->racket proc) v))] + [((~datum #%deforestable) (~datum filter-map) (proc:clause)) + #'(lambda (v) + (filter-map (qi0->racket proc) v))] + [((~datum #%deforestable) (~datum map) (proc:clause)) + #'(lambda (v) + (map (qi0->racket proc) v))] + [((~datum #%deforestable) (~datum foldl) (proc:clause) (init:expr)) + #'(lambda (v) + (foldl (qi0->racket proc) init v))] + [((~datum #%deforestable) (~datum foldr) (proc:clause) (init:expr)) + #'(lambda (v) + (foldr (qi0->racket proc) init v))] + [((~datum #%deforestable) (~datum range) () (arg:expr ...)) + #'(lambda () + (range arg ...))] + [((~datum #%deforestable) (~datum take) () (n:expr)) + #'(lambda (v) + (take v n))] + [((~datum #%deforestable) (~datum car)) + #'car] + [((~datum #%deforestable) (~datum cadr)) + #'cadr] + [((~datum #%deforestable) (~datum caddr)) + #'caddr] + [((~datum #%deforestable) (~datum cadddr)) + #'cadddr] + [((~datum #%deforestable) (~datum list-ref) () (n:expr)) + #'(lambda (v) + (list-ref v n))] + [((~datum #%deforestable) (~datum length)) + #'length] + [((~datum #%deforestable) (~or* (~datum empty?) (~datum null?))) + #'empty?])) + + (define (blanket-template-form-parser stx) + (syntax-parse stx + ;; "prarg" = "pre-supplied argument" + ;; Note: use of currying here doesn't play well with bindings + ;; because curry / curryr immediately evaluate their arguments + ;; and resolve any references to bindings at compile time. + ;; That's why we use a lambda which delays evaluation until runtime + ;; when the reference is actually resolvable. See "anaphoric references" + ;; in the compiler meeting notes, + ;; "The Artist Formerly Known as Bindingspec" + [((~datum #%blanket-template) + (natex prarg-pre ...+ (~datum __) prarg-post ...+)) + ;; "(curry (curryr ...) ...)" + #'(lambda largs + (apply + (lambda rargs + ((kw-helper natex rargs) prarg-post ...)) + prarg-pre ... + largs))] + [((~datum #%blanket-template) (natex prarg-pre ...+ (~datum __))) + ;; "curry" + #'(lambda args + (apply natex prarg-pre ... args))] + [((~datum #%blanket-template) + (natex (~datum __) prarg-post ...+)) + ;; "curryr" + #'(lambda args + ((kw-helper natex args) prarg-post ...))]))) diff --git a/qi-lib/flow/core/compiler/2000-bindings.rkt b/qi-lib/flow/core/compiler/2000-bindings.rkt new file mode 100644 index 000000000..7c51a53b4 --- /dev/null +++ b/qi-lib/flow/core/compiler/2000-bindings.rkt @@ -0,0 +1,73 @@ +#lang racket/base + +(require (for-syntax racket/base + syntax/parse + "../strategy.rkt") + racket/undefined + "../passes.rkt") + +;; Transformation rules for the `as` binding form: +;; +;; 1. escape to wrap outermost ~> with let and re-enter +;; +;; (~> flo ... (... (as name) ...)) +;; ... +;; ↓ +;; ... +;; (esc (let ([name (void)]) +;; (☯ original-flow))) +;; +;; 2. as → set! +;; +;; (as name) +;; ... +;; ↓ +;; ... +;; (~> (esc (λ (x) (set! name x))) ⏚) +;; +;; 3. Overall transformation: +;; +;; (~> flo ... (... (as name) ...)) +;; ... +;; ↓ +;; ... +;; (esc (let ([name (void)]) +;; (☯ (~> flo ... (... (~> (esc (λ (x) (set! name x))) ⏚) ...))))) + +(begin-for-syntax + + ;; (as name) → (~> (esc (λ (x) (set! name x))) ⏚) + ;; TODO: use a box instead of set! + (define (rewrite-all-bindings stx) + (find-and-map/qi (syntax-parser + [((~datum as) x ...) + #:with (x-val ...) (generate-temporaries (attribute x)) + #'(thread (esc (λ (x-val ...) (set! x x-val) ...)) ground)] + [_ this-syntax]) + stx)) + + (define (bound-identifiers stx) + (let ([ids null]) + (find-and-map/qi (syntax-parser + [((~datum as) x ...) + (begin + (set! ids + (append (attribute x) ids)) + ;; we don't need to traverse further + #f)] + [_ this-syntax]) + stx) + ids)) + + ;; wrap stx with (let ([v undefined] ...) stx) for v ∈ ids + (define (wrap-with-scopes stx ids) + (with-syntax ([(v ...) ids]) + #`(let ([v undefined] ...) #,stx))) + + (define-and-register-pass 2000 (bindings stx) + ;; TODO: use syntax-parse and match ~> specifically. + ;; Since macros are expanded "outside in," presumably + ;; it will naturally wrap the outermost ~> + (wrap-with-scopes (rewrite-all-bindings stx) + (bound-identifiers stx)))) + diff --git a/qi-lib/flow/core/compiler/deforest/cps.rkt b/qi-lib/flow/core/compiler/deforest/cps.rkt new file mode 100644 index 000000000..59ccc7d94 --- /dev/null +++ b/qi-lib/flow/core/compiler/deforest/cps.rkt @@ -0,0 +1,381 @@ +#lang racket/base + +(provide (for-syntax deforest-pass)) + +(require (for-syntax racket/base + syntax/parse + "syntax.rkt" + "../../../extended/util.rkt" + syntax/srcloc + racket/syntax-srcloc + "fusion.rkt" + "../../private/form-property.rkt") + "templates.rkt" + racket/performance-hint + racket/match + racket/contract/base) + +;; "Composes" higher-order functions inline by directly applying them +;; to the result of each subsequent application, with the last argument +;; being passed to the penultimate application as a (single) argument. +;; This is specialized to our implementation of stream fusion in the +;; arguments it expects and how it uses them. +(define-syntax inline-compose1 + (syntax-rules () + [(_ f) f] + [(_ [op (f ...) g ...] rest ...) (op f ... (inline-compose1 rest ...) g ...)] + )) + +;; Adds the initial states of all stateful transformers in the +;; required order to the initial producer state. Uses (cons Tx S) +;; where Tx is the transformer's initial state and S is the producer's +;; initial state with all preceding transformer states already +;; added. Nothing is added for stateless transformers which pass () as +;; their initial state expression. For example: (inline-consing (T1) +;; () (T2) P) -> (cons T2 (cons T1 P)) +(define-syntax inline-consing + (syntax-rules () + [(_ state () rest ...) (inline-consing state rest ...)] + [(_ state (arg) rest ...) (inline-consing (cons arg state) rest ...)] + [(_ state) state] + )) + +(begin-for-syntax + + ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; + ;; Producers + + ;; Special "curry"ing for #%fine-templates. All #%host-expressions are + ;; passed as they are and all (~datum _) are replaced by wrapper + ;; lambda arguments. + (define ((make-fine-curry argstx minargs maxargs form-stx) ctx name) + (define argstxlst (syntax->list argstx)) + (define numargs (length argstxlst)) + (cond + [(< numargs minargs) + (raise-syntax-error (syntax->datum name) + (format "too few arguments - given ~a - accepts at least ~a" + numargs minargs) + (prettify-flow-syntax ctx) + (prettify-flow-syntax form-stx))] + [(> numargs maxargs) + (raise-syntax-error (syntax->datum name) + (format "too many arguments - given ~a - accepts at most ~a" + numargs maxargs) + (prettify-flow-syntax ctx) + (prettify-flow-syntax form-stx))]) + (define temporaries (generate-temporaries argstxlst)) + (define-values (allargs tmpargs) + (for/fold ([all '()] + [tmps '()] + #:result (values (reverse all) + (reverse tmps))) + ([tmp (in-list temporaries)] + [arg (in-list argstxlst)]) + (syntax-parse arg + #:datum-literals (#%host-expression) + [(#%host-expression ex) + (values (cons #'ex all) + tmps)] + [(~datum _) + (values (cons tmp all) + (cons tmp tmps))]))) + (with-syntax ([(carg ...) tmpargs] + [(aarg ...) allargs]) + #'(lambda (proc) + (lambda (carg ...) + (proc aarg ...))))) + + ;; Special curry for #%blanket-template. Raises syntax error if there + ;; are too many arguments. If the number of arguments is exactly the + ;; maximum, wraps into lambda without any arguments. If less than + ;; maximum, curries it from both left and right. + (define ((make-blanket-curry prestx poststx maxargs form-stx) ctx name) + (define prelst (syntax->list prestx)) + (define postlst (syntax->list poststx)) + (define numargs (+ (length prelst) (length postlst))) + (with-syntax ([(pre-arg ...) prelst] + [(post-arg ...) postlst]) + (cond + [(> numargs maxargs) + (raise-syntax-error (syntax->datum name) + (format "too many arguments - given ~a - accepts at most ~a" + numargs maxargs) + (prettify-flow-syntax ctx) + (prettify-flow-syntax form-stx))] + [(= numargs maxargs) + #'(lambda (v) + (lambda () + (v pre-arg ... post-arg ...)))] + [else + #'(lambda (v) + (lambda rest + (apply v pre-arg ... + (append rest + (list post-arg ...)))))]))) + ;; Unifying producer curry makers. The ellipsis escaping allows for + ;; simple specification of pattern variable names as bound in the + ;; syntax pattern. + (define-syntax make-producer-curry + (syntax-rules () + [(_ min-args max-args + blanket? pre-arg post-arg + fine? arg + form-stx) + (cond + [(attribute blanket?) + (make-blanket-curry pre-arg + post-arg + max-args + #'form-stx + )] + [(attribute fine?) + (make-fine-curry arg min-args max-args #'form-stx)] + [else + (lambda (ctx name) #'(lambda (v) v))])])) + + (define-syntax-class fsp + #:attributes (curry name contract prepare next) + (pattern range:fsp-range + #:attr name #''range + #:attr contract #'(->* (real?) (real? real?) any) + #:attr prepare #'range->cstream-prepare + #:attr next #'range->cstream-next + #:attr curry (make-producer-curry 1 3 + range.blanket? #'range.pre-arg #'range.post-arg + range.fine? #'range.arg + range)) + (pattern default:fsp-default + #:attr name #''list->cstream + #:attr contract #'(-> list? any) + #:attr prepare #'list->cstream-prepare + #:attr next #'list->cstream-next + #:attr curry (lambda (ctx name) #'(lambda (v) v))) + ) + + ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; + ;; Transformers + + (define-syntax-class fst + #:attributes (next f state) + (pattern filter:fst-filter + #:attr f #'(filter.f) + #:attr next #'filter-cstream-next + #:attr state #'()) + (pattern map:fst-map + #:attr f #'(map.f) + #:attr next #'map-cstream-next + #:attr state #'()) + (pattern filter-map:fst-filter-map + #:attr f #'(filter-map.f) + #:attr next #'filter-map-cstream-next + #:attr state #'()) + (pattern take:fst-take + #:attr f #'() + #:attr next #'take-cstream-next + #:attr state #'(take.n)) + ) + + ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; + ;; Consumers + + (define-syntax-class fsc + #:attributes (end) + (pattern foldr:fsc-foldr + #:attr end #'(foldr-cstream-next foldr.op foldr.init)) + (pattern foldl:fsc-foldl + #:attr end #'(foldl-cstream-next foldl.op foldl.init)) + (pattern list-ref:fsc-list-ref + #:attr end #'(list-ref-cstream-next list-ref.pos 'list-ref.name)) + (pattern length:fsc-length + #:attr end #'(length-cstream-next)) + (pattern empty?:fsc-empty? + #:attr end #'(empty?-cstream-next)) + (pattern default:fsc-default + #:attr end #'(cstream-next->list)) + ) + + ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; + ;; The pass + + ;; Performs deforestation rewrite on the whole syntax tree. + (define-and-register-deforest-pass (deforest-pass ops ctx) + (syntax-parse (reverse ops) + [(c:fsc + t:fst ... + p:fsp) + ;; A static runtime contract is placed at the beginning of the + ;; fused sequence. And runtime checks for consumers are in + ;; their respective implementation procedure. + (with-syntax (((rt ...) (reverse (attribute t.state)))) + (attach-form-property + #`(esc + (#,((attribute p.curry) ctx (attribute p.name)) + (contract p.contract + (p.prepare + (lambda (state) + (define cstate (inline-consing state rt ...)) + cstate) + (#,@#'c.end + (inline-compose1 [t.next t.f + '#,(prettify-flow-syntax ctx) + '#,(build-source-location-vector + (syntax-srcloc ctx)) + ] ... + p.next + ) + '#,(prettify-flow-syntax ctx) + '#,(build-source-location-vector + (syntax-srcloc ctx)))) + p.name + '#,(prettify-flow-syntax ctx) + #f + '#,(build-source-location-vector + (syntax-srcloc ctx)))))))])) + + ) + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; Runtime + +(begin-encourage-inline + + ;; Producers + + (define-inline (list->cstream-next done skip yield) + (λ (state) + (cond [(null? state) (done)] + [else (yield (car state) (cdr state))]))) + + (define-inline (list->cstream-prepare consing next) + (case-lambda + [(lst) (next (consing lst))] + [rest (void)])) + + (define-inline (range->cstream-next done skip yield) + (λ (state) + (match-define (list l h s) state) + (cond [(< l h) + (yield l (cons (+ l s) (cdr state)))] + [else (done)]))) + + (define-inline (range->cstream-prepare consing next) + (case-lambda + [(h) (next (consing (list 0 h 1)))] + [(l h) (next (consing (list l h 1)))] + [(l h s) (next (consing (list l h s)))] + [rest (void)])) + + ;; Transformers + + (define-inline (map-cstream-next f next ctx src) + (λ (done skip yield) + (next done + skip + (λ (value state) + (yield (f value) state))))) + + (define-inline (filter-cstream-next f next ctx src) + (λ (done skip yield) + (next done + skip + (λ (value state) + (if (f value) + (yield value state) + (skip state)))))) + + (define-inline (filter-map-cstream-next f next ctx src) + (λ (done skip yield) + (next done + skip + (λ (value state) + (let ([fv (f value)]) + (if fv + (yield fv state) + (skip state))))))) + + (define-inline (take-cstream-next next ctx src) + (λ (done skip yield) + (λ (take-state) + (define n (car take-state)) + (define state (cdr take-state)) + (if (zero? n) + (done) + ((next (λ () + ((contract (-> pair? any) + (λ (v) v) + 'take ctx + #f + src) + '())) + (λ (state) + (skip (cons n state))) + (λ (value state) + (define new-state (cons (sub1 n) state)) + (yield value new-state))) + state))))) + + ;; Consumers + + (define-inline (cstream-next->list next ctx src) + (λ (state) + (let loop ([state state]) + ((next (λ () null) + (λ (state) (loop state)) + (λ (value state) + (cons value (loop state)))) + state)))) + + (define-inline (foldr-cstream-next op init next ctx src) + (λ (state) + (let loop ([state state]) + ((next (λ () init) + (λ (state) (loop state)) + (λ (value state) + (op value (loop state)))) + state)))) + + (define-inline (foldl-cstream-next op init next ctx src) + (λ (state) + (let loop ([acc init] [state state]) + ((next (λ () acc) + (λ (state) (loop acc state)) + (λ (value state) + (loop (op value acc) state))) + state)))) + + (define-inline (list-ref-cstream-next init-countdown name next ctx src) + (λ (state) + (let loop ([state state] + [countdown init-countdown]) + ((next (λ () ((contract (-> pair? any) + (λ (v) v) + name ctx #f + src) + '())) + (λ (state) (loop state countdown)) + (λ (value state) + (if (zero? countdown) + value + (loop state (sub1 countdown))))) + state)))) + + (define-inline (length-cstream-next next ctx src) + (λ (state) + (let loop ([state state] + [the-length 0]) + ((next (λ () the-length) + (λ (state) (loop state the-length)) + (λ (value state) + (loop state (add1 the-length)))) + state)))) + + (define-inline (empty?-cstream-next next ctx src) + (λ (state) + (let loop ([state state]) + ((next (λ () #t) + (λ (state) (loop state)) + (λ (value state) #f)) + state)))) + + ) diff --git a/qi-lib/flow/core/compiler/deforest/fusion.rkt b/qi-lib/flow/core/compiler/deforest/fusion.rkt new file mode 100644 index 000000000..46448995c --- /dev/null +++ b/qi-lib/flow/core/compiler/deforest/fusion.rkt @@ -0,0 +1,80 @@ +#lang racket/base + +(provide define-and-register-deforest-pass) + +(require (for-syntax racket/base + syntax/parse) + syntax/parse + "syntax.rkt" + "../../passes.rkt" + "../../strategy.rkt" + (for-template "../../passes.rkt") + "../../private/form-property.rkt") + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; The actual fusion generator implementation + +;; Used only in deforest-rewrite to properly recognize the end of +;; fusable sequence. +(define-syntax-class non-fusable + (pattern (~not (~or _:fst-syntax + _:fsp-syntax + _:fsc-syntax)))) + +(define (make-deforest-rewrite generate-fused-operation) + (lambda (stx) + (attach-form-property + (syntax-parse stx + [((~datum thread) _0:non-fusable ... + p:fsp-syntax + ;; There can be zero transformers here: + t:fst-syntax ... + c:fsc-syntax + _1 ...) + #:with fused (generate-fused-operation + (syntax->list #'(p t ... c)) + stx) + #'(thread _0 ... fused _1 ...)] + [((~datum thread) _0:non-fusable ... + t1:fst-syntax0 + t:fst-syntax ... + c:fsc-syntax + _1 ...) + #:with fused (generate-fused-operation + (syntax->list #'(list->cstream t1 t ... c)) + stx) + #'(thread _0 ... fused _1 ...)] + [((~datum thread) _0:non-fusable ... + p:fsp-syntax + ;; Must be 1 or more transformers here: + t:fst-syntax ...+ + _1 ...) + #:with fused (generate-fused-operation + (syntax->list #'(p t ... cstream->list)) + stx) + #'(thread _0 ... fused _1 ...)] + [((~datum thread) _0:non-fusable ... + f1:fst-syntax0 + f:fst-syntax ...+ + _1 ...) + #:with fused (generate-fused-operation + (syntax->list #'(list->cstream f1 f ... cstream->list)) + stx) + #'(thread _0 ... fused _1 ...)] + ;; return the input syntax unchanged if no rules + ;; are applicable + [_ stx])))) + +;; This syntax is actively used only once as it is intended to be used +;; by alternative implementations. Currently only the CPS +;; implementation uses it, however in the near future the named-let +;; implementation will use it as well. +(define-syntax (define-and-register-deforest-pass stx) + (syntax-parse stx + ((_ (deforest-pass ops ctx) expr ...) + #'(define-and-register-pass 100 (deforest-pass stx) + (find-and-map/qi + (make-deforest-rewrite + (lambda (ops ctx) + expr ...)) + stx))))) diff --git a/qi-lib/flow/core/compiler/deforest/syntax.rkt b/qi-lib/flow/core/compiler/deforest/syntax.rkt new file mode 100644 index 000000000..405c81578 --- /dev/null +++ b/qi-lib/flow/core/compiler/deforest/syntax.rkt @@ -0,0 +1,184 @@ +#lang racket/base + +(provide fsp-syntax + fst-syntax0 + fst-syntax + fsc-syntax + + fsp-range + fsp-default + + fst-filter + fst-map + fst-filter-map + fst-take + + fsc-foldr + fsc-foldl + fsc-list-ref + fsc-length + fsc-empty? + fsc-default + + ) + +(require syntax/parse + "../../passes.rkt" + "../../strategy.rkt" + (for-template racket/base + "../../passes.rkt" + "../../strategy.rkt" + "templates.rkt") + (for-syntax racket/base + syntax/parse)) + +;; Literals set used for matching Fusable Stream Literals +(define-literal-set fs-literals + #:datum-literals (esc #%host-expression #%fine-template #%blanket-template _ __) + ()) + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; Fusable Stream Producers +;; +;; Syntax classes used for matching functions that produce a sequence +;; of values and they annotate the syntax with attributes that will be +;; used in the compiler to apply optimizations. +;; +;; All are prefixed with fsp- for clarity. + +(define-syntax-class fsp-range + #:attributes (blanket? fine? arg pre-arg post-arg) + #:literal-sets (fs-literals) + #:datum-literals (range) + (pattern (#%deforestable range () (the-arg ...)) + #:attr arg #'(the-arg ...) + #:attr pre-arg #f + #:attr post-arg #f + #:attr blanket? #f + #:attr fine? #t)) + +(define-syntax-class fsp-default + #:datum-literals (list->cstream) + (pattern list->cstream + #:attr contract #'(-> list? any) + #:attr name #''list->cstream)) + +(define-syntax-class fsp-syntax + (pattern (~or _:fsp-range + _:fsp-default))) + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; Fusable Stream Transformers +;; +;; Syntax classes matching functions acting as transformers of the +;; sequence of values passing through. +;; +;; All are prefixed with fst- for clarity. + +(define-syntax-class fst-filter + #:attributes (f) + #:literal-sets (fs-literals) + #:datum-literals (filter) + (pattern (#%deforestable filter (f-uncompiled)) + #:attr f (run-passes #'f-uncompiled))) +(define-syntax-class fst-map + #:attributes (f) + #:literal-sets (fs-literals) + #:datum-literals (map) + (pattern (#%deforestable map (f-uncompiled)) + #:attr f (run-passes #'f-uncompiled))) + +(define-syntax-class fst-filter-map + #:attributes (f) + #:literal-sets (fs-literals) + #:datum-literals (filter-map) + (pattern (#%deforestable filter-map (f-uncompiled)) + #:attr f (run-passes #'f-uncompiled))) + +(define-syntax-class fst-take + #:attributes (n) + #:literal-sets (fs-literals) + #:datum-literals (take) + (pattern (#%deforestable take () ((#%host-expression n))))) + +(define-syntax-class fst-syntax0 + (pattern (~or _:fst-filter + _:fst-filter-map))) + +(define-syntax-class fst-syntax + (pattern (~or _:fst-filter + _:fst-map + _:fst-filter-map + _:fst-take))) + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; Fusable Stream Consumers +;; +;; Syntax classes used for matching functions that can consume all +;; values from a sequence and create a single value from those. +;; +;; Prefixed with fsc- for clarity. + +(define-syntax-class fsc-foldr + #:attributes (op init) + #:literal-sets (fs-literals) + #:datum-literals (foldr) + (pattern (#%deforestable + foldr + (op-uncompiled) + ((#%host-expression init))) + #:attr op (run-passes #'op-uncompiled))) + +(define-syntax-class fsc-foldl + #:attributes (op init) + #:literal-sets (fs-literals) + #:datum-literals (foldl) + (pattern (#%deforestable + foldl + (op-uncompiled) + ((#%host-expression init))) + #:attr op (run-passes #'op-uncompiled))) + +(define-syntax-class cad*r-datum + #:attributes (countdown) + (pattern (#%deforestable (~datum car)) #:attr countdown #'0) + (pattern (#%deforestable (~datum cadr)) #:attr countdown #'1) + (pattern (#%deforestable (~datum caddr)) #:attr countdown #'2) + (pattern (#%deforestable (~datum cadddr)) #:attr countdown #'3)) + +(define-syntax-class fsc-list-ref + #:attributes (pos name) + #:literal-sets (fs-literals) + #:datum-literals (list-ref) + ;; TODO: need #%host-expression wrapping idx? + (pattern (#%deforestable list-ref () (idx)) + #:attr pos #'idx + #:attr name #'list-ref) + ;; TODO: bring wrapping #%deforestable out here? + (pattern cad*r:cad*r-datum + #:attr pos #'cad*r.countdown + #:attr name #'cad*r)) + +(define-syntax-class fsc-length + #:literal-sets (fs-literals) + #:datum-literals (length) + (pattern (#%deforestable length))) + +(define-syntax-class fsc-empty? + #:literal-sets (fs-literals) + #:datum-literals (null? empty?) + (pattern (#%deforestable (~or empty? + null?)))) + +(define-syntax-class fsc-default + #:datum-literals (cstream->list) + (pattern cstream->list)) + +(define-syntax-class fsc-syntax + (pattern (~or _:fsc-foldr + _:fsc-foldl + _:fsc-list-ref + _:fsc-length + _:fsc-empty? + _:fsc-default + ))) diff --git a/qi-lib/flow/core/compiler/deforest/templates.rkt b/qi-lib/flow/core/compiler/deforest/templates.rkt new file mode 100644 index 000000000..9f9038bc2 --- /dev/null +++ b/qi-lib/flow/core/compiler/deforest/templates.rkt @@ -0,0 +1,10 @@ +#lang racket/base + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; Special syntax for templates + +;; These bindings are used for ~literal matching to introduce implicit +;; producer/consumer when none is explicitly given in the flow. +(provide cstream->list list->cstream) +(define cstream->list #'-cstream->list) +(define list->cstream #'-list->cstream) diff --git a/qi-lib/flow/core/deforest.rkt b/qi-lib/flow/core/deforest.rkt deleted file mode 100644 index 71e35896a..000000000 --- a/qi-lib/flow/core/deforest.rkt +++ /dev/null @@ -1,422 +0,0 @@ -#lang racket/base - -;; This module implements the stream fusion optimization to "deforest" -;; sequences of functional transformations (e.g. map, filter, fold, etc.) -;; so that they avoid constructing intermediate representations on the -;; way to producing the final result. -;; -;; See the wiki -;; https://github.com/drym-org/qi/wiki/The-Compiler#stream-fusion -;; for an overview and some details of this implementation. - -(provide (for-syntax deforest-pass)) - -(require (for-syntax racket/base - syntax/parse - racket/syntax-srcloc - syntax/srcloc - "../extended/util.rkt" - "strategy.rkt") - "passes.rkt" - racket/performance-hint - racket/match - racket/list - racket/contract/base) - -;; These bindings are used for ~literal matching to introduce implicit -;; producer/consumer when none is explicitly given in the flow. -(define cstream->list #'-cstream->list) -(define list->cstream #'-list->cstream) - -;; "Composes" higher-order functions inline by directly applying them -;; to the result of each subsequent application, with the last argument -;; being passed to the penultimate application as a (single) argument. -;; This is specialized to our implementation of stream fusion in the -;; arguments it expects and how it uses them. -(define-syntax inline-compose1 - (syntax-rules () - [(_ f) f] - [(_ [op f] rest ...) (op f (inline-compose1 rest ...))])) - -(begin-for-syntax - ;; Special "curry"ing for #%fine-templates. All #%host-expressions - ;; are passed as they are and all (~datum _) are replaced by wrapper - ;; lambda arguments. - (define ((make-fine-curry argstx minargs maxargs form-stx) ctx name) - (define argstxlst (syntax->list argstx)) - (define numargs (length argstxlst)) - (cond - [(< numargs minargs) - (raise-syntax-error (syntax->datum name) - (format "too few arguments - given ~a - accepts at least ~a" - numargs minargs) - (prettify-flow-syntax ctx) - (prettify-flow-syntax form-stx))] - [(> numargs maxargs) - (raise-syntax-error (syntax->datum name) - (format "too many arguments - given ~a - accepts at most ~a" - numargs maxargs) - (prettify-flow-syntax ctx) - (prettify-flow-syntax form-stx))]) - (define temporaries (generate-temporaries argstxlst)) - (define-values (allargs tmpargs) - (for/fold ([all '()] - [tmps '()] - #:result (values (reverse all) - (reverse tmps))) - ([tmp (in-list temporaries)] - [arg (in-list argstxlst)]) - (syntax-parse arg - #:datum-literals (#%host-expression) - [(#%host-expression ex) - (values (cons #'ex all) - tmps)] - [(~datum _) - (values (cons tmp all) - (cons tmp tmps))]))) - (with-syntax ([(carg ...) tmpargs] - [(aarg ...) allargs]) - #'(λ (proc) - (λ (carg ...) - (proc aarg ...))))) - - ;; Special curry for #%blanket-template. Raises syntax error if - ;; there are too many arguments. If the number of arguments is - ;; exactly the maximum, wraps into lambda without any arguments. If - ;; less than maximum, curries it from both left and right. - (define ((make-blanket-curry prestx poststx maxargs form-stx) ctx name) - (define prelst (syntax->list prestx)) - (define postlst (syntax->list poststx)) - (define numargs (+ (length prelst) (length postlst))) - (with-syntax ([(pre-arg ...) prelst] - [(post-arg ...) postlst]) - (cond - [(> numargs maxargs) - (raise-syntax-error (syntax->datum name) - (format "too many arguments - given ~a - accepts at most ~a" - numargs maxargs) - (prettify-flow-syntax ctx) - (prettify-flow-syntax form-stx))] - [(= numargs maxargs) - #'(λ (v) - (λ () - (v pre-arg ... post-arg ...)))] - [else - #'(λ (v) - (λ rest - (apply v pre-arg ... - (append rest - (list post-arg ...)))))]))) - - ;; Unifying producer curry makers. The ellipsis escaping allows for - ;; simple specification of pattern variable names as bound in the - ;; syntax pattern. - (define-syntax make-producer-curry - (syntax-rules () - [(_ min-args max-args - blanket? pre-arg post-arg - fine? arg - form-stx) - (cond - [(attribute blanket?) - (make-blanket-curry #'(pre-arg (... ...)) - #'(post-arg (... ...)) - max-args - #'form-stx - )] - [(attribute fine?) - (make-fine-curry #'(arg (... ...)) min-args max-args #'form-stx)] - [else - (λ (ctx name) #'(λ (v) v))])])) - - ;; Used for producing the stream from particular - ;; expressions. Implicit producer is list->cstream-next and it is - ;; not created by using this class but rather explicitly used when - ;; no syntax class producer is matched. - (define-syntax-class fusable-stream-producer - #:attributes (next prepare contract name curry) - #:datum-literals (#%host-expression #%blanket-template #%fine-template esc __) - ;; Explicit range producers. - (pattern (~and (~or (esc (#%host-expression (~datum range))) - (~and (#%fine-template - ((#%host-expression (~datum range)) - arg ...)) - fine?) - (~and (#%blanket-template - ((#%host-expression (~datum range)) - (#%host-expression pre-arg) ... - __ - (#%host-expression post-arg) ...)) - blanket?)) - form-stx) - #:attr next #'range->cstream-next - #:attr prepare #'range->cstream-prepare - #:attr contract #'(->* (real?) (real? real?) any) - #:attr name #'range - #:attr curry (make-producer-curry 1 3 - blanket? pre-arg post-arg - fine? arg - form-stx)) - - ;; The implicit stream producer from plain list. - (pattern (~literal list->cstream) - #:attr next #'list->cstream-next - #:attr prepare #'list->cstream-prepare - #:attr contract #'(-> list? any) - #:attr name #''list->cstream - #:attr curry (λ (ctx name) #'(λ (v) v)))) - - ;; Matches any stream transformer that can be in the head position - ;; of the fused sequence even when there is no explicit - ;; producer. Procedures accepting variable number of arguments like - ;; `map` cannot be in this class. - (define-syntax-class fusable-stream-transformer0 - #:attributes (f next) - #:datum-literals (#%host-expression #%blanket-template __ _ #%fine-template) - (pattern (~or (#%blanket-template - ((#%host-expression (~datum filter)) - (#%host-expression f) - __)) - (#%fine-template - ((#%host-expression (~datum filter)) - (#%host-expression f) - _))) - #:attr next #'filter-cstream-next)) - - ;; All implemented stream transformers - within the stream, only - ;; single value is being passed and therefore procedures like `map` - ;; can (and should) be matched. - (define-syntax-class fusable-stream-transformer - #:attributes (f next) - #:datum-literals (#%host-expression #%blanket-template __ _ #%fine-template) - (pattern (~or (#%blanket-template - ((#%host-expression (~datum map)) - (#%host-expression f) - __)) - (#%fine-template - ((#%host-expression (~datum map)) - (#%host-expression f) - _))) - #:attr next #'map-cstream-next) - - (pattern (~or (#%blanket-template - ((#%host-expression (~datum filter)) - (#%host-expression f) - __)) - (#%fine-template - ((#%host-expression (~datum filter)) - (#%host-expression f) - _))) - #:attr next #'filter-cstream-next)) - - ;; Terminates the fused sequence (consumes the stream) and produces - ;; an actual result value. - (define-syntax-class fusable-stream-consumer - #:attributes (end) - #:datum-literals (#%host-expression #%blanket-template _ __ #%fine-template esc) - (pattern (~or (#%blanket-template - ((#%host-expression (~datum foldr)) - (#%host-expression op) - (#%host-expression init) - __)) - (#%fine-template - ((#%host-expression (~datum foldr)) - (#%host-expression op) - (#%host-expression init) - _))) - #:attr end #'(foldr-cstream-next op init)) - - (pattern (~or (#%blanket-template - ((#%host-expression (~datum foldl)) - (#%host-expression op) - (#%host-expression init) - __)) - (#%fine-template - ((#%host-expression (~datum foldl)) - (#%host-expression op) - (#%host-expression init) - _))) - #:attr end #'(foldl-cstream-next op init)) - - (pattern (~or (esc (#%host-expression (~datum car))) - (#%fine-template - ((#%host-expression (~datum car)) - _)) - (#%blanket-template - ((#%host-expression (~datum car)) - __))) - #:attr end #'(car-cstream-next)) - - (pattern (~literal cstream->list) - #:attr end #'(cstream-next->list))) - - ;; Used only in deforest-rewrite to properly recognize the end of - ;; fusable sequence. - (define-syntax-class non-fusable - (pattern (~not (~or _:fusable-stream-transformer - _:fusable-stream-producer - _:fusable-stream-consumer)))) - - ;; Generates a syntax for the fused operation for given - ;; sequence. The syntax list must already be in the following form: - ;; (producer transformer ... consumer) - (define (generate-fused-operation ops ctx) - (syntax-parse (reverse ops) - [(c:fusable-stream-consumer - t:fusable-stream-transformer ... - p:fusable-stream-producer) - ;; A static runtime contract is placed at the beginning of the - ;; fused sequence. And runtime checks for consumers are in - ;; their respective implementation procedure. - #`(esc - (#,((attribute p.curry) ctx (attribute p.name)) - (contract p.contract - (p.prepare - (#,@#'c.end - (inline-compose1 [t.next t.f] ... - p.next) - '#,(prettify-flow-syntax ctx) - '#,(build-source-location-vector - (syntax-srcloc ctx)))) - p.name - '#,(prettify-flow-syntax ctx) - #f - '#,(build-source-location-vector - (syntax-srcloc ctx)))))])) - - (define (deforest-rewrite stx) - (syntax-parse stx - [((~datum thread) _0:non-fusable ... - p:fusable-stream-producer - ;; There can be zero transformers here: - t:fusable-stream-transformer ... - c:fusable-stream-consumer - _1 ...) - #:with fused (generate-fused-operation - (syntax->list #'(p t ... c)) - stx) - #'(thread _0 ... fused _1 ...)] - [((~datum thread) _0:non-fusable ... - t1:fusable-stream-transformer0 - t:fusable-stream-transformer ... - c:fusable-stream-consumer - _1 ...) - #:with fused (generate-fused-operation - (syntax->list #'(list->cstream t1 t ... c)) - stx) - #'(thread _0 ... fused _1 ...)] - [((~datum thread) _0:non-fusable ... - p:fusable-stream-producer - ;; Must be 1 or more transformers here: - t:fusable-stream-transformer ...+ - _1 ...) - #:with fused (generate-fused-operation - (syntax->list #'(p t ... cstream->list)) - stx) - #'(thread _0 ... fused _1 ...)] - [((~datum thread) _0:non-fusable ... - f1:fusable-stream-transformer0 - f:fusable-stream-transformer ...+ - _1 ...) - #:with fused (generate-fused-operation - (syntax->list #'(list->cstream f1 f ... cstream->list)) - stx) - #'(thread _0 ... fused _1 ...)] - ;; return the input syntax unchanged if no rules - ;; are applicable - [_ stx])) - - ;; Performs deforestation rewrite on the whole syntax tree. - (define-and-register-pass 100 (deforest-pass stx) - (find-and-map/qi - (fix deforest-rewrite) - stx))) - -(begin-encourage-inline - - ;; Producers - - (define-inline (list->cstream-next done skip yield) - (λ (state) - (cond [(null? state) (done)] - [else (yield (car state) (cdr state))]))) - - (define-inline (list->cstream-prepare next) - (case-lambda - [(lst) (next lst)] - [rest (void)])) - - (define-inline (range->cstream-next done skip yield) - (λ (state) - (match-define (list l h s) state) - (cond [(< l h) - (yield l (cons (+ l s) (cdr state)))] - [else (done)]))) - - (define-inline (range->cstream-prepare next) - (case-lambda - [(h) (next (list 0 h 1))] - [(l h) (next (list l h 1))] - [(l h s) (next (list l h s))] - [rest (void)])) - - ;; Transformers - - (define-inline (map-cstream-next f next) - (λ (done skip yield) - (next done - skip - (λ (value state) - (yield (f value) state))))) - - (define-inline (filter-cstream-next f next) - (λ (done skip yield) - (next done - skip - (λ (value state) - (if (f value) - (yield value state) - (skip state)))))) - - ;; Consumers - - (define-inline (cstream-next->list next ctx src) - (λ (state) - (let loop ([state state]) - ((next (λ () null) - (λ (state) (loop state)) - (λ (value state) - (cons value (loop state)))) - state)))) - - (define-inline (foldr-cstream-next op init next ctx src) - (λ (state) - (let loop ([state state]) - ((next (λ () init) - (λ (state) (loop state)) - (λ (value state) - (op value (loop state)))) - state)))) - - (define-inline (foldl-cstream-next op init next ctx src) - (λ (state) - (let loop ([acc init] [state state]) - ((next (λ () acc) - (λ (state) (loop acc state)) - (λ (value state) - (loop (op value acc) state))) - state)))) - - (define-inline (car-cstream-next next ctx src) - (λ (state) - (let loop ([state state]) - ((next (λ () ((contract (-> pair? any) - (λ (v) v) - 'car-cstream-next ctx #f - src) '())) - (λ (state) (loop state)) - (λ (value state) - value)) - state)))) - - ) diff --git a/qi-lib/flow/core/syntax.rkt b/qi-lib/flow/core/syntax.rkt index 2cf8a0ca8..e70ff09e3 100644 --- a/qi-lib/flow/core/syntax.rkt +++ b/qi-lib/flow/core/syntax.rkt @@ -17,7 +17,8 @@ fold-left-form fold-right-form loop-form - clos-form) + clos-form + deforestable-form) (require syntax/parse) @@ -26,10 +27,13 @@ These syntax classes are used in the flow macro to handle matching of the input syntax to valid Qi syntax. Typically, _matching_ is the only function these syntax classes fulfill, and once matched, the input syntax is typically handed over to dedicated parsers that -independently parse and expand the input. It's done this way to keep -the clauses of the flow macro specific to individual forms, instead of -these forms appearing in multiple clauses, so that the code for each -form is decoupled from the rest of the flow macro. +independently parse and expand the input. It's done this way for two +reasons. First, the syntax has already been parsed/validated by the +expander and we don't need to worry about validation at the compiler +level. And second, to keep the clauses of the Qi0→Racket codegen macro +specific to individual forms, instead of these forms appearing in +multiple clauses, so that the code for each form is neatly decoupled +from code generation for other forms. See comments in flow.rkt for more details. |# @@ -131,3 +135,7 @@ See comments in flow.rkt for more details. (~datum clos)) (pattern ((~datum clos) arg ...))) + +(define-syntax-class deforestable-form + (pattern + ((~datum #%deforestable) arg ...))) diff --git a/qi-lib/flow/extended/expander.rkt b/qi-lib/flow/extended/expander.rkt index 3f09b836e..3215ddf44 100644 --- a/qi-lib/flow/extended/expander.rkt +++ b/qi-lib/flow/extended/expander.rkt @@ -183,6 +183,11 @@ core language's use of #%app, etc.). (clos onex:closed-floe) (esc ex:racket-expr) + ;; core form to express deforestable operations + (#%deforestable name:id (f:closed-floe ...) (arg:racket-expr ...)) + (#%deforestable name:id (f:closed-floe ...+)) + (#%deforestable name:id) + ;; backwards compat macro extensibility via Racket macros (~> ((~var ext-form (starts-with "qi:")) expr ...) #'(esc (ext-form expr ...))) diff --git a/qi-lib/list.rkt b/qi-lib/list.rkt index dd3e56b3a..0d96362d3 100644 --- a/qi-lib/list.rkt +++ b/qi-lib/list.rkt @@ -1,8 +1,64 @@ #lang racket/base -;; Upon instantiation of the module it define-and-register-pass for -;; deforestation -(require racket/list - "flow/core/deforest.rkt") +(provide (for-space qi + (all-defined-out))) -(provide (all-from-out racket/list)) +(require (for-syntax racket/base + "private/util.rkt") + syntax/parse/define + "flow/extended/expander.rkt" + (only-in "flow/space.rkt" + define-qi-alias) + "macro.rkt") + +(define-qi-syntax-rule (map f:expr) + (#%deforestable map (f))) + +(define-qi-syntax-rule (filter f:expr) + (#%deforestable filter (f))) + +(define-qi-syntax-rule (filter-map f:expr) + (#%deforestable filter-map (f))) + +(define-qi-syntax-rule (foldl f:expr init:expr) + (#%deforestable foldl (f) (init))) + +(define-qi-syntax-rule (foldr f:expr init:expr) + (#%deforestable foldr (f) (init))) + +(define-qi-syntax-parser range + [(_ low:expr high:expr step:expr) #'(#%deforestable range () (low high step))] + [(_ low:expr high:expr) #'(#%deforestable range () (low high 1))] + [(_ high:expr) #'(#%deforestable range () (0 high 1))] + ;; not strictly necessary but this provides a better error + ;; message than simply "range: bad syntax" that's warranted + ;; to differentiate from racket/list's `range` + [_:id (report-syntax-error this-syntax + "(range arg ...)" + "range expects at least one argument")]) + +(define-qi-syntax-rule (take n:expr) + (#%deforestable take () (n))) + +(define-qi-syntax-parser car + [_:id #'(#%deforestable car)]) + +(define-qi-syntax-parser cadr + [_:id #'(#%deforestable cadr)]) + +(define-qi-syntax-parser caddr + [_:id #'(#%deforestable caddr)]) + +(define-qi-syntax-parser cadddr + [_:id #'(#%deforestable cadddr)]) + +(define-qi-syntax-rule (list-ref n:expr) + (#%deforestable list-ref () (n))) + +(define-qi-syntax-parser length + [_:id #'(#%deforestable length)]) + +(define-qi-syntax-parser empty? + [_:id #'(#%deforestable empty?)]) + +(define-qi-alias null? empty?) diff --git a/qi-lib/private/util.rkt b/qi-lib/private/util.rkt index 4a6c329aa..35023f5fc 100644 --- a/qi-lib/private/util.rkt +++ b/qi-lib/private/util.rkt @@ -24,6 +24,18 @@ "" (string-append "\n" (string-join msgs "\n")))) + stx)] + [name + (raise-syntax-error name + (~a "Syntax error in " + name + "\n" + "Usage:\n" + " " usage + (if (null? msgs) + "" + (string-append "\n" + (string-join msgs "\n")))) stx)])) (define-syntax-parse-rule (define-alias alias:id name:id) diff --git a/qi-sdk/benchmarks/nonlocal/qi/main.rkt b/qi-sdk/benchmarks/nonlocal/qi/main.rkt index b02a16040..8bf6ad414 100644 --- a/qi-sdk/benchmarks/nonlocal/qi/main.rkt +++ b/qi-sdk/benchmarks/nonlocal/qi/main.rkt @@ -18,7 +18,6 @@ double-values) (require (only-in math sqr) - (only-in racket/list range) qi qi/list) @@ -43,8 +42,9 @@ [else (~> (-< sub1 (- 2)) (>< pingala) +)]) -(define-flow (eratosthenes n) - (~> (-< (gen null) (~>> add1 (range 2) △)) +(define (eratosthenes n) + (~> () + (-< (gen null) (~>> (range 2 (add1 n)) △)) (feedback (while (~> (block 1) live?)) (then (~> 1> reverse)) (-< (~> (select 1 2) X cons) @@ -77,24 +77,26 @@ (map sqr) (foldl + 0))) -(define-flow range-map-car - (~>> (range 0) +(define (range-map-car n) + (~>> () + (range 0 n) (map sqr) car)) -(define-flow range-map-sum +(define (range-map-sum n) ;; TODO: this should be written as (apply +) ;; and that should be normalized to (foldr/l + 0) ;; (depending on which of foldl/foldr is more performant) - (~>> (range 0) (map sqr) (foldr + 0))) + (~>> () (range 0 n) (map sqr) (foldr + 0))) -(define-flow long-functional-pipeline - (~>> (range 0) +(define (long-functional-pipeline n) + (~>> () + (range 0 n) (filter odd?) (map sqr) values - (filter (λ (v) (< (remainder v 10) 5))) - (map (λ (v) (* 2 v))) + (filter (~> (remainder 10) (< 5))) + (map (* 2)) (foldl + 0))) ;; (define filter-double diff --git a/qi-test/tests/compiler.rkt b/qi-test/tests/compiler.rkt index a132f4f92..0c2e7f8de 100644 --- a/qi-test/tests/compiler.rkt +++ b/qi-test/tests/compiler.rkt @@ -4,7 +4,6 @@ (require rackunit rackunit/text-ui - (prefix-in semantics: "compiler/semantics.rkt") (prefix-in rules: "compiler/rules.rkt") (prefix-in strategy: "compiler/strategy.rkt") (prefix-in impl: "compiler/impl.rkt")) @@ -13,7 +12,6 @@ (test-suite "compiler tests" - semantics:tests rules:tests strategy:tests impl:tests)) diff --git a/qi-test/tests/compiler/rules/deforest.rkt b/qi-test/tests/compiler/rules/deforest.rkt index 5d2f06704..6e67fe033 100644 --- a/qi-test/tests/compiler/rules/deforest.rkt +++ b/qi-test/tests/compiler/rules/deforest.rkt @@ -9,7 +9,8 @@ ;; necessary to correctly expand the right-threading form qi/flow/extended/forms qi/flow/core/compiler - qi/flow/core/deforest + qi/flow/core/compiler/0100-deforest + qi/list syntax/macro-testing (submod qi/flow/extended/expander invoke) rackunit @@ -33,132 +34,178 @@ (deforest-pass (expand-flow stx)))))) +(define-syntax-parse-rule (test-deforest stx) + (phase1-eval + (deforest-pass + (expand-flow stx)))) + (define tests (test-suite "deforestation" ;; Note that these test deforestation in isolation - ;; without necessarily taking normalization (a preceding - ;; step in compilation) into account + ;; *without* taking normalization (a preceding + ;; step in compilation) into account. + ;; If a test is failing that you are expecting should pass, + ;; it could be that it implicitly assumes that normalization + ;; will be done, so double check this. + ;; For testing behavior of the full cycle of compilation + ;; involving normalization as well as deforestation, use the + ;; `full-cycle.rkt` test module. (test-suite "deforest-pass" + (test-suite "general" (test-not-deforested "does not deforest single stream component in isolation" - #'(~>> (filter odd?))) + #'(~> (filter odd?))) (test-not-deforested "does not deforest map in the head position" - #'(~>> (map sqr) (filter odd?))) - (test-deforested "deforestation in arbitrary positions" + #'(~> (map sqr) (filter odd?))) + (test-deforested "deforestation is invariant to threading direction" #'(~>> values (filter odd?) (map sqr) values)) (test-deforested "deforestation in arbitrary positions" - #'(~>> - values - (filter string-upcase) - (foldl string-append "I") - values)) + #'(~> values + (filter odd?) + (map sqr) + values)) + (test-deforested "deforestation in arbitrary positions" + #'(~> values + (filter string-upcase) + (foldl string-append "I") + values)) ;; TODO: this test is for a case where deforestation should be applied twice ;; to the same expression. But currently, the test does not differentiate ;; between the optimization being applied once vs twice. We would like it ;; to do so in order to validate and justify the need for fixed-point ;; finding in the deforestation pass. (test-deforested "multiple applications of deforestation to the same expression" - #'(~>> (filter odd?) - (map sqr) - (foldr + 0) - range - (filter odd?) - (map sqr)))) + #'(~> (filter odd?) + (map sqr) + (foldr + 0) + (as v) + (range v) + (filter odd?) + (map sqr))) + (test-true "nested positions" + (deforested? (phase1-eval + (deforest-pass + (expand-flow + #'(>< (~> (filter odd?) (map sqr)))))))) + (test-case "multiple independent positions" + (let ([stx (phase1-eval + (deforest-pass + (expand-flow + #'(-< (~> (filter odd?) (map sqr)) + (~> (as v) (range v) car)))))]) + (check-true (deforested? stx)) + (check-true (filter-deforested? stx)) + (check-true (list-ref-deforested? stx))))) (test-suite "transformers" - (test-deforested "filter-map (two transformers)" - #'(~>> (filter odd?) (map sqr))) - (test-deforested "fine-grained template forms" - #'(~>> (filter odd? _) (map sqr _)))) + (test-deforested "filter->map (two transformers)" + #'(~> (filter odd?) (map sqr))) + (test-suite + "filter" + (test-true "filter" + (filter-deforested? + (test-deforest + #'(~> (filter odd?) (map sqr)))))) + (test-suite + "map" + (test-true "map" + (map-deforested? + (test-deforest + #'(~> (filter odd?) (map sqr)))))) + (test-suite + "filter-map" + (test-true "filter-map" + (filter-map-deforested? + (test-deforest + #'(~> (filter odd?) (filter-map sqr)))))) + (test-suite + "take" + (test-true "take" + (take-deforested? + (test-deforest + #'(~> (filter odd?) (take 3))))))) (test-suite "producers" - ;; TODO: note that these uses of `range` are matched as datums - ;; and requiring racket/list's range is not required in this module - ;; for deforestation to happen. This should be changed to use - ;; literal matching in the compiler. - (test-deforested "range" - #'(~>> range (filter odd?))) - (test-deforested "(range _)" - #'(~>> (range _) (filter odd?))) - (test-deforested "(range _ _)" - #'(~>> (range _ _) (filter odd?))) - (test-deforested "(range 0 _)" - #'(~>> (range 0 _) (filter odd?))) - (test-deforested "(range _ 10)" - #'(~>> (range _ 10) (filter odd?))) - (test-deforested "(range _ _ _)" - #'(~>> (range _ _ _) (filter odd?))) - (test-deforested "(range _ _ 1)" - #'(~>> (range _ _ 1) (filter odd?))) - (test-deforested "(range _ 10 _)" - #'(~>> (range _ 10 _) (filter odd?))) - (test-deforested "(range _ 10 1)" - #'(~>> (range _ 10 1) (filter odd?))) - (test-deforested "(range 0 _ _)" - #'(~>> (range 0 _ _) (filter odd?))) - (test-deforested "(range 0 _ 1)" - #'(~>> (range 0 _ 1) (filter odd?))) - (test-deforested "(range 0 10 _)" - #'(~>> (range 0 10 _) (filter odd? __))) - (test-deforested "(range __)" - #'(~>> (range __) (filter odd?))) - (test-deforested "(range 0 __)" - #'(~>> (range 0 __) (filter odd?))) - (test-deforested "(range __ 1)" - #'(~>> (range __ 1) (filter odd?))) - (test-deforested "(range 0 10 __)" - #'(~>> (range 0 10 __) (filter odd?))) - (test-deforested "(range __ 10 1)" - #'(~>> (range __ 10 1) (filter odd? __))) - (test-deforested "(range 0 __ 1)" - #'(~>> (range 0 __ 1) (filter odd?))) - (test-deforested "(range 0 10 1 __)" - #'(~>> (range 0 10 1 __) (filter odd?))) - (test-deforested "(range 0 10 __ 1)" - #'(~>> (range 0 10 __ 1) (filter odd?))) - (test-deforested "(range 0 __ 10 1)" - #'(~>> (range 0 __ 10 1) (filter odd?))) - (test-deforested "(range __ 0 10 1)" - #'(~>> (range __ 0 10 1) (filter odd?)))) + (test-suite + "range" + (test-true "range" + (range-deforested? + (test-deforest + #'(~> (range 10) (filter odd?))))) + (test-true "range" + (range-deforested? + (test-deforest + #'(~> (range 1 10) (filter odd?))))) + (test-true "range" + (range-deforested? + (test-deforest + #'(~> (range 1 10 2) (filter odd?))))))) (test-suite "consumers" - (test-deforested "car" - #'(~>> (filter odd?) car)) - (test-deforested "foldl" - #'(~>> (filter string-upcase) (foldl string-append "I"))) - (test-deforested "foldr" - #'(~>> (filter string-upcase) (foldr string-append "I"))))) - - (test-suite - "deforest-pass" - (test-true "nested positions" - (deforested? (phase1-eval - (deforest-pass - (expand-flow - #'(>< (~>> (filter odd?) (map sqr)))))))) - (let ([stx (phase1-eval - (deforest-pass - (expand-flow - #'(-< (~>> (filter odd?) (map sqr)) - (~>> range car)))))]) - (test-true "multiple independent positions" - (deforested? stx)) - (test-true "multiple independent positions" - (filter-deforested? stx)) - (test-true "multiple independent positions" - (car-deforested? stx)))))) + (test-suite + "list-ref" + (test-deforested "car" + #'(~> (filter odd?) car)) + (test-true "car" + (list-ref-deforested? + (test-deforest + #'(~> (filter odd?) car)))) + (test-deforested "list-ref" + #'(~> (filter odd?) (list-ref 2))) + (test-true "list-ref" + (list-ref-deforested? + (test-deforest + #'(~> (filter odd?) (list-ref 2)))))) + (test-suite + "foldl" + (test-deforested "foldl" + #'(~> (filter non-empty-string?) (foldl string-append "I"))) + (test-true "foldl" + (foldl-deforested? + (test-deforest + #'(~> (filter non-empty-string?) (foldl string-append "I")))))) + (test-suite + "foldr" + (test-deforested "foldr" + #'(~> (filter non-empty-string?) (foldr string-append "I"))) + (test-true "foldr" + (foldr-deforested? + (test-deforest + #'(~> (filter non-empty-string?) (foldr string-append "I")))))) + (test-suite + "length" + (test-deforested "length" + #'(~> (filter non-empty-string?) length)) + (test-true "length" + (length-deforested? + (test-deforest + #'(~> (filter non-empty-string?) length))))) + (test-suite + "empty?" + (test-deforested "empty?" + #'(~> (filter non-empty-string?) empty?)) + (test-true "empty?" + (empty?-deforested? + (test-deforest + #'(~> (filter non-empty-string?) empty?)))) + (test-deforested "null?" + #'(~> (filter non-empty-string?) null?)) + (test-true "null?" + (empty?-deforested? + (test-deforest + #'(~> (filter non-empty-string?) null?))))))))) (module+ main (void diff --git a/qi-test/tests/compiler/rules/full-cycle.rkt b/qi-test/tests/compiler/rules/full-cycle.rkt index e7e174966..4ef5f615f 100644 --- a/qi-test/tests/compiler/rules/full-cycle.rkt +++ b/qi-test/tests/compiler/rules/full-cycle.rkt @@ -10,8 +10,8 @@ rackunit rackunit/text-ui syntax/macro-testing - qi/flow/core/deforest qi/flow/core/compiler + qi/list "private/deforest-util.rkt" (submod qi/flow/extended/expander invoke)) @@ -33,7 +33,18 @@ (deforested? (phase1-eval (qi-compile - #'(~>> (filter odd?) values (map sqr))))))))) + #'(~>> (filter odd?) values (map sqr)))))) + ;; We expect the Qi expander to translate threading direction simply to + ;; chirality of individual contained forms (indicated by the presence of + ;; a blanket template on either side) if they are applications of host + ;; language functions, and to leave them unchanged if they are syntactic + ;; forms (where chirality is irrelevant). We also expect normalization + ;; to collapse nested threading forms, so that the following should be + ;; deforested. + (test-true "nested, different threading direction" + (deforested? (phase1-eval + (qi-compile + #'(~> (filter odd?) (~>> (map sqr)))))))))) (module+ main (void diff --git a/qi-test/tests/compiler/rules/private/deforest-util.rkt b/qi-test/tests/compiler/rules/private/deforest-util.rkt index 193a986d8..0b7d0230b 100644 --- a/qi-test/tests/compiler/rules/private/deforest-util.rkt +++ b/qi-test/tests/compiler/rules/private/deforest-util.rkt @@ -1,8 +1,16 @@ #lang racket/base (provide deforested? + range-deforested? filter-deforested? - car-deforested?) + map-deforested? + filter-map-deforested? + take-deforested? + foldl-deforested? + foldr-deforested? + length-deforested? + empty?-deforested? + list-ref-deforested?) ;; Note: an alternative way to make these assertions could be to add logging ;; to compiler passes to trace what happens to a source expression, capturing @@ -16,8 +24,32 @@ (define (deforested? exp) (string-contains? (format "~a" exp) "cstream")) +(define (range-deforested? exp) + (string-contains? (format "~a" exp) "range->cstream")) + (define (filter-deforested? exp) (string-contains? (format "~a" exp) "filter-cstream")) -(define (car-deforested? exp) - (string-contains? (format "~a" exp) "car-cstream")) +(define (map-deforested? exp) + (string-contains? (format "~a" exp) "map-cstream")) + +(define (filter-map-deforested? exp) + (string-contains? (format "~a" exp) "filter-map-cstream")) + +(define (take-deforested? exp) + (string-contains? (format "~a" exp) "take-cstream")) + +(define (foldl-deforested? exp) + (string-contains? (format "~a" exp) "foldl-cstream")) + +(define (foldr-deforested? exp) + (string-contains? (format "~a" exp) "foldr-cstream")) + +(define (length-deforested? exp) + (string-contains? (format "~a" exp) "length-cstream")) + +(define (empty?-deforested? exp) + (string-contains? (format "~a" exp) "empty?-cstream")) + +(define (list-ref-deforested? exp) + (string-contains? (format "~a" exp) "list-ref-cstream")) diff --git a/qi-test/tests/compiler/semantics.rkt b/qi-test/tests/compiler/semantics.rkt deleted file mode 100644 index 2ffcad815..000000000 --- a/qi-test/tests/compiler/semantics.rkt +++ /dev/null @@ -1,189 +0,0 @@ -#lang racket/base - -(provide tests) - -(require qi - rackunit - rackunit/text-ui - (only-in math sqr) - qi/list - syntax/macro-testing - racket/function) - -(define tests - (test-suite - "Compiler preserves semantics" - - (test-suite - "deforestation" - - (test-suite - "general" - (check-equal? ((☯ (~>> (filter odd?) (map sqr))) - (list 1 2 3 4 5)) - (list 1 9 25)) - (check-exn exn:fail? - (thunk - ((☯ (~> (map sqr) (map sqr))) - (list 1 2 3 4 5))) - "(map) doforestation should only be done for right threading") - (check-exn exn:fail? - (thunk - ((☯ (~> (filter odd?) (filter odd?))) - (list 1 2 3 4 5))) - "(filter) doforestation should only be done for right threading") - (check-exn exn:fail? - (thunk - ((☯ (~>> (filter odd?) (~> (foldr + 0)))) - (list 1 2 3 4 5))) - "(foldr) doforestation should only be done for right threading") - (check-equal? ((☯ (~>> values (filter odd?) (map sqr) values)) - (list 1 2 3 4 5)) - (list 1 9 25) - "optimizes subexpressions") - (check-equal? ((☯ (~>> (filter odd?) (map sqr) (foldr + 0))) - (list 1 2 3 4 5)) - 35) - (check-equal? ((☯ (~>> (filter odd?) (map sqr) (foldl + 0))) - (list 1 2 3 4 5)) - 35) - (check-equal? ((☯ (~>> (map string-upcase) (foldr string-append "I"))) - (list "a" "b" "c")) - "ABCI") - (check-equal? ((☯ (~>> (map string-upcase) (foldl string-append "I"))) - (list "a" "b" "c")) - "CBAI") - (check-equal? ((☯ (~>> (range 10) (map sqr) car))) - 0)) - - (test-suite - "error reporting" - (test-exn "deforestation syntax phase - too many arguments for range producer (blanket)" - exn? - (lambda () - (convert-compile-time-error - ((flow (~>> (range 1 2 3 4 5) (filter odd?) (map sqr))))))) - - (test-exn "deforestation syntax phase - too many arguments for range producer (fine)" - exn? - (lambda () - (convert-compile-time-error - ((flow (~>> (range 1 2 3 4 5 _) (filter odd?) (map sqr))))))) - - (test-equal? "deforestation list->cstream-next usage" - ((flow (~>> (filter odd?) (map sqr))) - '(0 1 2 3 4 5 6 7 8 9)) - '(1 9 25 49 81)) - - (test-exn "deforestation range->cstream-next - too few arguments at runtime" - exn? - (lambda () - ((flow (~>> range (filter odd?) (map sqr)))))) - - (test-exn "deforestation range->cstream-next - too many arguments at runtime" - exn? - (lambda () - ((flow (~>> range (filter odd?) (map sqr))) 1 2 3 4))) - - (test-exn "deforestation car-cstream-next - empty list" - exn? - (lambda () - ((flow (~>> (filter odd?) (map sqr) car)) '())))) - - (test-suite - "range (stream producer)" - ;; Semantic tests of the range producer that cover all combinations: - (test-equal? "~>>range [1-3] (10)" - (~>> (10) range (filter odd?) (map sqr)) - '(1 9 25 49 81)) - (test-equal? "~>range [1-3] (10)" - (~> (10) range (~>> (filter odd?) (map sqr))) - '(1 9 25 49 81)) - (test-equal? "~>> range [1-3] (5 10)" - (~>> (5 10) range (filter odd?) (map sqr)) - '(25 49 81)) - (test-equal? "~> range [1-3] (5 10)" - (~> (5 10) range (~>> (filter odd?) (map sqr))) - '(25 49 81)) - (test-equal? "~>> range [1-3] (5 10 3)" - (~>> (5 10 3) range (filter odd?) (map sqr)) - '(25)) - (test-equal? "~> range [1-3] (5 10 3)" - (~> (5 10 3) range (~>> (filter odd?) (map sqr))) - '(25)) - - (test-equal? "~>> (range 10) [0-2] ()" - (~>> () (range 10) (filter odd?) (map sqr)) - '(1 9 25 49 81)) - (test-equal? "~> (range 10) [0-2] ()" - (~> () (range 10) (~>> (filter odd?) (map sqr))) - '(1 9 25 49 81)) - (test-equal? "~>> (range 5) [0-2] (10)" - (~>> (10) (range 5) (filter odd?) (map sqr)) - '(25 49 81)) - (test-equal? "~> (range 10) [0-2] (5)" - (~> (5) (range 10) (~>> (filter odd?) (map sqr))) - '(25 49 81)) - (test-equal? "~>> (range 3) [0-2] (5 10)" - (~>> (3) (range 5 10) (filter odd?) (map sqr)) - '(25)) - (test-equal? "~> (range 5) [0-2] (10 3)" - (~> (5) (range 10 3) (~>> (filter odd?) (map sqr))) - '(25)) - - (test-equal? "~>> (range 5 10) [0-1] ()" - (~>> () (range 5 10) (filter odd?) (map sqr)) - '(25 49 81)) - (test-equal? "~> (range 5 10) [0-1] ()" - (~> () (range 5 10) (~>> (filter odd?) (map sqr))) - '(25 49 81)) - (test-equal? "~>> (range 5 10) [0-1] (3)" - (~>> (3) (range 5 10) (filter odd?) (map sqr)) - '(25)) - (test-equal? "~> (range 10 3) [0-1] (5)" - (~> (5) (range 10 3) (~>> (filter odd?) (map sqr))) - '(25)) - - (test-equal? "~>> (range 5 10 3) [0] ()" - (~>> () (range 5 10 3) (filter odd?) (map sqr)) - '(25)) - - (test-equal? "~>> (range _) [1] (10)" - (~>> (10) (range _) (filter odd?) (map sqr)) - '(1 9 25 49 81)) - (test-equal? "~>> (range _ _) [2] (5 10)" - (~>> (5 10) (range _ _) (filter odd?) (map sqr)) - '(25 49 81)) - (test-equal? "~>> (range _ _ _) [3] (5 10 3)" - (~>> (5 10 3) (range _ _ _) (filter odd?) (map sqr)) - '(25)) - - (test-equal? "~>> (range 5 _) [1] (10)" - (~>> (10) (range 5 _) (filter odd?) (map sqr)) - '(25 49 81)) - (test-equal? "~>> (range _ 10) [1] (5)" - (~>> (5) (range _ 10) (filter odd?) (map sqr)) - '(25 49 81)) - - (test-equal? "~>> (range 5 _ _) [2] (10 3)" - (~>> (10 3) (range 5 _ _) (filter odd?) (map sqr)) - '(25)) - (test-equal? "~>> (range _ 10 _) [2] (5 3)" - (~>> (5 3) (range _ 10 _) (filter odd?) (map sqr)) - '(25)) - (test-equal? "~>> (range _ _ 3) [2] (5 10)" - (~>> (5 10) (range _ _ 3) (filter odd?) (map sqr)) - '(25)) - - (test-equal? "~>> (range 5 10 _) [1] (3)" - (~>> (3) (range 5 10 _) (filter odd?) (map sqr)) - '(25)) - (test-equal? "~>> (range 5 _ 3) [1] (10)" - (~>> (10) (range 5 _ 3) (filter odd?) (map sqr)) - '(25)) - (test-equal? "~>> (range _ 10 3) [1] (5)" - (~>> (5) (range _ 10 3) (filter odd?) (map sqr)) - '(25)))))) - -(module+ main - (void (run-tests tests))) diff --git a/qi-test/tests/expander.rkt b/qi-test/tests/expander.rkt index 07e556920..76b99c3de 100644 --- a/qi-test/tests/expander.rkt +++ b/qi-test/tests/expander.rkt @@ -103,7 +103,12 @@ #'(thread (#%blanket-template ((#%host-expression f) (#%host-expression 1) - __))))) + __)))) + (test-expand "#%deforestable" + #'(#%deforestable name (_) (_)) + #'(#%deforestable name + (_) + ((#%host-expression _))))) (test-suite "utils" ;; this is just temporary until we properly track source expressions through diff --git a/qi-test/tests/flow.rkt b/qi-test/tests/flow.rkt index 5218767a7..3d14115cf 100644 --- a/qi-test/tests/flow.rkt +++ b/qi-test/tests/flow.rkt @@ -285,6 +285,16 @@ (check-equal? ((☯ (esc (first (list + *)))) 3 7) 10 "normal racket expressions")) + (test-suite + "#%deforestable" + (check-equal? ((☯ (#%deforestable filter (odd?))) (list 1 2 3)) (list 1 3)) + (check-equal? ((☯ (#%deforestable map (sqr))) (list 1 2 3)) (list 1 4 9)) + (check-equal? ((☯ (#%deforestable foldl (+) (0))) (list 1 2 3)) 6) + (check-equal? ((☯ (#%deforestable foldr (+) (0))) (list 1 2 3)) 6) + (check-equal? ((☯ (#%deforestable range () (3)))) (list 0 1 2)) + (check-equal? ((☯ (#%deforestable range () (0 3)))) (list 0 1 2)) + (check-equal? ((☯ (#%deforestable range () (0 5 2)))) (list 0 2 4)) + (check-equal? ((☯ (#%deforestable take () (2))) (list 1 2 3)) (list 1 2))) (test-suite "elementary boolean gates" (test-suite diff --git a/qi-test/tests/list.rkt b/qi-test/tests/list.rkt new file mode 100644 index 000000000..416df6b0f --- /dev/null +++ b/qi-test/tests/list.rkt @@ -0,0 +1,332 @@ +#lang racket/base + +(provide tests) + +(require qi + qi/list + rackunit + rackunit/text-ui + syntax/macro-testing + (only-in racket/function thunk) + (only-in racket/string non-empty-string?) + (only-in math sqr)) + +(define tests + (test-suite + "qi/list tests" + + (test-suite + "basic" + + (test-suite + "stream producers" + (test-suite + "range" + (test-equal? "single arg" + ((☯ (range 3))) + (list 0 1 2)) + (test-equal? "two args" + ((☯ (range 1 4))) + (list 1 2 3)) + (test-equal? "three args" + ((☯ (range 1 6 2))) + (list 1 3 5)) + (test-exn "expects at least one argument" + exn:fail:syntax? + (thunk + (convert-compile-time-error + ((☯ range))))))) + + (test-suite + "stream transformers" + (test-suite + "filter" + (test-equal? "simple list" + ((☯ (filter odd?)) + (list 1 2 3)) + (list 1 3)) + (test-equal? "empty list" + ((☯ (filter odd?)) + null) + null) + (test-equal? "no matching values" + ((☯ (filter odd?)) + (list 2 4 6)) + null) + (test-equal? "all matching values" + ((☯ (filter odd?)) + (list 1 3 5)) + (list 1 3 5)) + (test-equal? "filter with higher-order Qi syntax" + ((☯ (filter (and positive? integer?))) + (list 1 -2 3 0.2 4)) + (list 1 3 4))) + (test-suite + "map" + (test-equal? "simple list" + ((☯ (map sqr)) + (list 1 2 3)) + (list 1 4 9)) + (test-equal? "empty list" + ((☯ (map sqr)) + null) + null) + (test-equal? "map with higher-order Qi syntax" + ((☯ (map (~> sqr add1))) + (list 1 2 3)) + (list 2 5 10))) + (test-suite + "filter-map" + (test-equal? "simple list" + ((☯ (filter-map (if positive? sqr #false))) + (list 1 -2 3)) + (list 1 9)) + (test-equal? "empty list" + ((☯ (filter-map (if positive? sqr #false))) + null) + null)) + (test-suite + "take (stateful transformer)" + (test-equal? "simple list" + ((☯ (take 2)) + (list 1 2 3)) + (list 1 2)) + (test-equal? "take none" + ((☯ (take 0)) + (list 1 2 3)) + null) + (test-exn "empty list" + exn:fail:contract? + (thunk ((☯ (take 2)) + null))) + (test-equal? "take none from empty list" + ((☯ (take 0)) + null) + null))) + + (test-suite + "stream consumers" + (test-suite + "foldl" + (test-equal? "simple list" + ((☯ (foldl + 0)) + (list 1 2 3)) + 6) + (test-equal? "empty list" + ((☯ (foldl + 0)) + null) + 0) + (test-equal? "non-commutative operation" + ((☯ (foldl string-append "")) + (list "a" "b" "c")) + "cba") + (test-equal? "foldl with higher-order Qi syntax" + ((☯ (foldl (~> (>< number->string) + string-append + string->number) + 0)) + (list 1 2 3)) + 3210)) + (test-suite + "foldr" + (test-equal? "simple list" + ((☯ (foldr + 0)) + (list 1 2 3)) + 6) + (test-equal? "empty list" + ((☯ (foldr + 0)) + null) + 0) + (test-equal? "non-commutative operation" + ((☯ (foldr string-append "")) + (list "a" "b" "c")) + "abc") + (test-equal? "foldr with higher-order Qi syntax" + ((☯ (foldr (~> (>< number->string) + string-append + string->number) + 0)) + (list 1 2 3)) + 1230)) + (test-suite + "car" + (test-equal? "simple list" + ((☯ car) + (list 1 2 3)) + 1) + (test-exn "empty list" + exn:fail:contract? + (thunk ((☯ car) + null)))) + (test-suite + "null?" + (test-false "simple list" + ((☯ null?) + (list 1 2 3))) + (test-true "empty list" + ((☯ null?) + null))) + (test-suite + "empty?" + (test-false "simple list" + ((☯ empty?) + (list 1 2 3))) + (test-true "empty list" + ((☯ empty?) + null))) + (test-suite + "length" + (test-equal? "simple list" + ((☯ length) + (list 1 2 3)) + 3) + (test-equal? "empty list" + ((☯ length) + null) + 0)) + (test-suite + "list-ref" + (test-equal? "simple list" + ((☯ (list-ref 1)) + (list 1 2 3)) + 2) + (test-exn "empty list" + exn:fail:contract? + (thunk ((☯ (list-ref 1)) + null)))) + (test-suite + "cadr" + (test-equal? "simple list" + ((☯ cadr) + (list 1 2 3)) + 2) + (test-exn "empty list" + exn:fail:contract? + (thunk ((☯ cadr) + null)))) + (test-suite + "caddr" + (test-equal? "simple list" + ((☯ caddr) + (list 1 2 3)) + 3) + (test-exn "empty list" + exn:fail:contract? + (thunk ((☯ caddr) + null)))) + (test-suite + "cadddr" + (test-equal? "simple list" + ((☯ cadddr) + (list 1 2 3 4)) + 4) + (test-exn "empty list" + exn:fail:contract? + (thunk ((☯ cadddr) + null)))))) + + (test-suite + "combinations" + + (test-equal? "filter..map" + ((☯ (~> (filter odd?) + (map sqr))) + (list 1 2 3)) + (list 1 9)) + (test-equal? "filter..car" + ((☯ (~> (filter odd?) + car)) + (list 1 2 3)) + 1) + (test-equal? "filter..foldl" + ((☯ (~> (filter odd?) + (foldl + 0))) + (list 1 2 3)) + 4) + (test-equal? "filter..foldr" + ((☯ (~> (filter odd?) + (foldr + 0))) + (list 1 2 3)) + 4) + (test-equal? "filter..foldl with non-commutative operation" + ((☯ (~> (filter non-empty-string?) + (foldl string-append ""))) + (list "a" "b" "c")) + "cba") + (test-equal? "filter..foldr with non-commutative operation" + ((☯ (~> (filter non-empty-string?) + (foldr string-append ""))) + (list "a" "b" "c")) + "abc") + (test-equal? "map..foldl" + ((☯ (~> (map string-upcase) + (foldl string-append "I"))) + (list "a" "b" "c")) + "CBAI") + (test-equal? "map..foldr" + ((☯ (~> (map string-upcase) + (foldr string-append "I"))) + (list "a" "b" "c")) + "ABCI") + (test-equal? "range..car" + ((☯ (~> (range 10) + car))) + 0) + (test-equal? "range..map" + ((☯ (~> (range 3) + (map sqr)))) + (list 0 1 4)) + (test-equal? "range..filter..car" + ((☯ (~> (range 1 4) + (filter odd?) + car))) + 1) + (test-equal? "range..map..car" + ((☯ (~> (range 10) + (map sqr) + car))) + 0) + (test-equal? "filter..map..foldr" + ((☯ (~> (filter odd?) + (map sqr) + (foldr + 0))) + (list 1 2 3 4 5)) + 35) + (test-equal? "filter..map..foldl" + ((☯ (~> (filter odd?) + (map sqr) + (foldl + 0))) + (list 1 2 3 4 5)) + 35) + (test-equal? "range..filter..map" + ((☯ (~> (range 10) + (filter odd?) + (map sqr)))) + '(1 9 25 49 81)) + (test-equal? "range..filter..map with right threading" + ((☯ (~>> (range 10) + (filter odd?) + (map sqr)))) + '(1 9 25 49 81)) + (test-equal? "range..filter..map with different nested threading direction" + ((☯ (~> (range 10) + (~>> (filter odd?) + (map sqr))))) + '(1 9 25 49 81)) + (test-equal? "take after filter" + ((☯ (~> (range 20) + (filter odd?) + (take 5) + (map sqr)))) + '(1 9 25 49 81)) + (test-equal? "two takes after filter" + ((☯ (~> (range 20) + (filter odd?) + (take 5) + (take 3) + (map sqr)))) + '(1 9 25))))) + +(module+ main + (void + (run-tests tests))) diff --git a/qi-test/tests/qi.rkt b/qi-test/tests/qi.rkt index 7edbec6b0..5cbae34e6 100644 --- a/qi-test/tests/qi.rkt +++ b/qi-test/tests/qi.rkt @@ -11,7 +11,8 @@ (prefix-in macro: "macro.rkt") (prefix-in util: "util.rkt") (prefix-in expander: "expander.rkt") - (prefix-in compiler: "compiler.rkt")) + (prefix-in compiler: "compiler.rkt") + (prefix-in list: "list.rkt")) (define tests (test-suite @@ -26,7 +27,8 @@ macro:tests util:tests expander:tests - compiler:tests)) + compiler:tests + list:tests)) (module+ test (void diff --git a/qi-test/tests/util.rkt b/qi-test/tests/util.rkt index 9e0510a9c..0a628efeb 100644 --- a/qi-test/tests/util.rkt +++ b/qi-test/tests/util.rkt @@ -18,7 +18,10 @@ "blah: blah" "Use it" "like" - "this")))))) + "this"))) + (check-exn exn:fail:syntax? + (thunk (report-syntax-error #'dummy + "blah: blah")))))) (module+ main (void (run-tests tests)))