diff --git a/flexible-with.hl.rkt b/flexible-with.hl.rkt index 8d64c23..39b4d00 100644 --- a/flexible-with.hl.rkt +++ b/flexible-with.hl.rkt @@ -1,17 +1,17 @@ -#lang aful/unhygienic hyper-literate type-expander/lang +#lang hyper-literate #:♦ (dotlambda/unhygienic . type-expander/lang) -@title[#:style manual-doc-style ;#:style (with-html5 manual-doc-style) +♦title[#:style manual-doc-style ;#:style (with-html5 manual-doc-style) #:tag "flexible-with" #:tag-prefix "phc-graph/flexible-with"]{Flexible functional modification and extension of records} -@(chunks-toc-prefix +♦(chunks-toc-prefix '("(lib phc-graph/scribblings/phc-graph-implementation.scrbl)" "phc-graph/flexible-with")) -@section{Type of a tree-record, with a hole} +♦section{Type of a tree-record, with a hole} -@CHUNK[ +♦CHUNK[ (define-for-syntax (tree-type-with-replacement n last τ*) (define-values (next mod) (quotient/remainder n 2)) (cond [(null? τ*) last] @@ -24,23 +24,23 @@ #`(Pairof #,(car τ*) #,last) (cdr τ*))]))] -@section{Functionally updating a tree-record} +♦section{Functionally updating a tree-record} -@subsection{Adding and modifying fields} +♦subsection{Adding and modifying fields} Since we only deal with functional updates of immutable records, modifying a field does little more than discarding the old value, and injecting the new value instead into the new, updated record. Adding a new field is done using the same exact operation: missing fields are -denoted by a special value, @racket['NONE], while present fields are -represented as instances of the polymorphic struct @racket[(Some T)]. Adding a -new field is therefore as simple as discarding the old @racket['NONE] marker, -and replacing it with the new value, wrapped with @racket[Some]. A field -update would instead discard the old instance of @racket[Some], and replace it +denoted by a special value, ♦racket['NONE], while present fields are +represented as instances of the polymorphic struct ♦racket[(Some T)]. Adding a +new field is therefore as simple as discarding the old ♦racket['NONE] marker, +and replacing it with the new value, wrapped with ♦racket[Some]. A field +update would instead discard the old instance of ♦racket[Some], and replace it with a new one. -@CHUNK[ +♦CHUNK[ (if (= i 1) #'(delay/pure/stateless replacement) (let* ([bits (to-bits i)] @@ -55,7 +55,7 @@ with a new one. tree-thunk replacement))))] -@CHUNK[ +♦CHUNK[ (: replace-right (∀ (A B C R) (→ (→ (Promise B) R (Promise C)) (Promise (Pairof A B)) R @@ -131,12 +131,12 @@ with a new one. (define (rm-name tree-thunk) (low-name tree-thunk 'NONE))))] -@section{Auxiliary values} +♦section{Auxiliary values} The following sections reuse a few values which are derived from the list of fields: -@CHUNK[ +♦CHUNK[ (define all-fields #'(field …)) (define depth-above (ceiling-log2 (length (syntax->list #'(field …))))) (define offset (expt 2 depth-above)) @@ -157,9 +157,9 @@ fields: (stx-map (λ (f) (format-id f "u-with-~a" f)) #'(field …)))))] -@section{Type of a tree-record} +♦section{Type of a tree-record} -@CHUNK[<τ-tree-with-fields> +♦CHUNK[<τ-tree-with-fields> (define-for-syntax (τ-tree-with-fields struct-fields fields) (define/with-syntax (struct-field …) struct-fields) (define/with-syntax (field …) fields) @@ -171,7 +171,7 @@ fields: [i (in-naturals)]) (cons n (+ i offset))))) (define fields+indices - (sort (stx-map #λ(cons % (free-id-table-ref lookup %)) + (sort (stx-map λ.(cons % (free-id-table-ref lookup %)) #'(struct-field …)) < #:key cdr)) @@ -192,9 +192,9 @@ fields: ,(f (add1 (* i 2)))))))) (f 1))] -@section{Conversion to and from record-trees} +♦section{Conversion to and from record-trees} -@CHUNK[ +♦CHUNK[ (define-for-syntax (define-struct↔tree offset all-fields τ* struct-name fields) (define/with-syntax (field …) fields) @@ -208,7 +208,7 @@ fields: [i (in-naturals)]) (cons n (+ i offset))))) (define fields+indices - (sort (stx-map #λ(cons % (free-id-table-ref lookup %)) + (sort (stx-map λ.(cons % (free-id-table-ref lookup %)) fields) < #:key cdr)) @@ -229,9 +229,9 @@ fields: (define tree (force tree-thunk)) #,(convert-back-fields (* offset 2) fields+indices))))] -@subsection{Creating a new tree-record} +♦subsection{Creating a new tree-record} -@CHUNK[ +♦CHUNK[ (define-for-syntax (convert-fields up fields+indices) ;(displayln fields+indices) (define (f i) @@ -248,12 +248,12 @@ fields: (f 1))] -@subsection{Extracting all the fields from a tree-record} +♦subsection{Extracting all the fields from a tree-record} We traverse the tree in preorder, and accumulate definitions naming the interesting subparts of the trees (those where there are fields). -@CHUNK[ +♦CHUNK[ (define-for-syntax (convert-back-fields up fields+indices) (define result '()) (define definitions '()) @@ -292,20 +292,20 @@ interesting subparts of the trees (those where there are fields). (f 1 #'tree) #`(begin #,@definitions (values . #,(reverse result))))] -@section{Defining the converters and accessors for each known record type} +♦section{Defining the converters and accessors for each known record type} -@CHUNK[ +♦CHUNK[ (define-for-syntax (define-trees stx) (syntax-case stx () [(bt-fields-id (field …) [struct struct-field …] …) (let () - (define ∀-types (map #λ(format-id #'here "τ~a" %) + (define ∀-types (map λ.(format-id #'here "τ~a" %) (range (add1 depth-above)))) (define total-nb-functions (vector-length names)) )]))] -@CHUNK[ +♦CHUNK[ (define-for-syntax (bt-fields-type fields) (λ (stx) (syntax-case stx () @@ -313,27 +313,27 @@ interesting subparts of the trees (those where there are fields). #`(∀ fs (Promise #,(τ-tree-with-fields #'fs fields)))])))] -@CHUNK[ +♦CHUNK[ #`(begin (define-type-expander bt-fields-id (bt-fields-type #'#,(syntax-local-introduce #'(field …)))) - #,@(map #λ(define-replace-in-tree low-names + #,@(map λ.(define-replace-in-tree low-names names rm-names ∀-types % (floor-log2 %)) (range 1 (add1 total-nb-functions))) - #;#,@(map #λ(define-remove-in-tree rm-names ∀-types % (floor-log2 %)) + #;#,@(map λ.(define-remove-in-tree rm-names ∀-types % (floor-log2 %)) (range 1 (add1 total-nb-functions))) - #,@(map #λ(define-struct↔tree + #,@(map λ.(define-struct↔tree offset all-fields ∀-types %1 %2) (syntax->list #'(struct …)) (syntax->list #'([struct-field …] …))))] -@subsection{Putting it all together} +♦subsection{Putting it all together} -@chunk[ +♦chunk[ (struct (T) Some ([v : T]) #:transparent) (define-type (Maybe T) (U (Some T) 'NONE))] -@chunk[<*> +♦chunk[<*> (require delay-pure "flexible-with-utils.hl.rkt" (for-syntax (rename-in racket/base [... …]) @@ -363,4 +363,4 @@ interesting subparts of the trees (those where there are fields). ] -@include-section[(submod "flexible-with-utils.hl.rkt" doc)] \ No newline at end of file +♦include-section[(submod "flexible-with-utils.hl.rkt" doc)] \ No newline at end of file diff --git a/test/invariant-phantom/simple.rkt b/test/invariant-phantom/simple.rkt index 97872b9..51a9ae2 100644 --- a/test/invariant-phantom/simple.rkt +++ b/test/invariant-phantom/simple.rkt @@ -140,4 +140,5 @@ (:a ≡ :a.f.h) (:a ∉ :a.f.g)) (Invariants (:a.l ≥ (+ (length :a.f.g) 2)) - (:a ≢ :a.f.g.x))) + (:a ≢ :a.f.g.0) + (:a ≢ :a.f.g.1))) diff --git a/test/test-flexible-with.rkt b/test/test-flexible-with.rkt index c4ab572..9f571bc 100644 --- a/test/test-flexible-with.rkt +++ b/test/test-flexible-with.rkt @@ -1,4 +1,4 @@ -#lang aful/unhygienic type-expander/lang +#lang dotlambda/unhygienic type-expander/lang (require (lib "phc-graph/flexible-with.hl.rkt") (for-syntax racket/syntax @@ -43,37 +43,37 @@ (check-equal?: (call-with-values - #λ(tree→sab (sab→tree 1 2)) + λ.(tree→sab (sab→tree 1 2)) list) '(1 2)) (check-equal?: (call-with-values - #λ(tree→sabc (ann (with-c (sab→tree 1 2) 'nine) - ((bt-fields a b c) One Positive-Byte 'nine))) + λ.(tree→sabc (ann (with-c (sab→tree 1 2) 'nine) + ((bt-fields a b c) One Positive-Byte 'nine))) list) '(1 2 nine)) (check-equal?: (call-with-values - #λ(tree→sabc (with-c (sab→tree 'NONE 'NONE) 'NONE)) + λ.(tree→sabc (with-c (sab→tree 'NONE 'NONE) 'NONE)) list) '(NONE NONE NONE)) (check-equal?: (call-with-values - #λ(tree→sab (without-c (with-c (sab→tree 'NONE 'NONE) 'NONE))) + λ.(tree→sab (without-c (with-c (sab→tree 'NONE 'NONE) 'NONE))) list) '(NONE NONE)) (check-equal?: (call-with-values - #λ(tree→sbc (without-a (with-c (sab→tree 'NONE 'NONE) 'NONE))) + λ.(tree→sbc (without-a (with-c (sab→tree 'NONE 'NONE) 'NONE))) list) '(NONE NONE)) (check-equal?: (call-with-values - #λ(tree→sbc (without-a (with-c (sab→tree 1 2) 3))) + λ.(tree→sbc (without-a (with-c (sab→tree 1 2) 3))) list) '(2 3)) \ No newline at end of file