Preliminary support for tagged structures, early draft for unions
This commit is contained in:
parent
4cc991e751
commit
4862573453
24
dispatch-union.rkt
Normal file
24
dispatch-union.rkt
Normal file
|
@ -0,0 +1,24 @@
|
||||||
|
#lang typed/racket
|
||||||
|
|
||||||
|
(require phc-toolkit
|
||||||
|
(for-syntax racket/base
|
||||||
|
phc-toolkit/untyped
|
||||||
|
racket/syntax
|
||||||
|
syntax/parse
|
||||||
|
syntax/parse/experimental/template
|
||||||
|
type-expander/expander
|
||||||
|
"free-identifier-tree-equal.rkt")
|
||||||
|
(for-meta 2 racket/base)
|
||||||
|
(for-meta 2 phc-toolkit/untyped)
|
||||||
|
(for-meta 2 syntax/parse))
|
||||||
|
|
||||||
|
(define-syntax/parse (dispatch-union ([type-to-replaceᵢ Aᵢ predicateᵢ] …)
|
||||||
|
[X v result] …)
|
||||||
|
(stx-map
|
||||||
|
(λ (X v result)
|
||||||
|
(cond
|
||||||
|
[(meta-struct? X) #`[((struct-predicate #,X) #,v) #,result]]
|
||||||
|
[else (raise-syntax-error 'graph "Unhandled union type" #'X)]))
|
||||||
|
#'(X …)
|
||||||
|
#'(v …)
|
||||||
|
#'(result …)))
|
2
test/adt-pre-declarations.rkt
Normal file
2
test/adt-pre-declarations.rkt
Normal file
|
@ -0,0 +1,2 @@
|
||||||
|
#lang s-exp phc-adt/declarations
|
||||||
|
(remembered! tagged-structure (tg a b))
|
|
@ -10,11 +10,11 @@
|
||||||
|
|
||||||
(define-syntax check-equal?-values:
|
(define-syntax check-equal?-values:
|
||||||
(syntax-parser
|
(syntax-parser
|
||||||
[(_ actual {~maybe :colon type} expected ...)
|
[(_ actual {~maybe :colon type:type-expand!} expected ...)
|
||||||
(quasisyntax/top-loc this-syntax
|
(quasisyntax/top-loc this-syntax
|
||||||
(check-equal?: (call-with-values (ann (λ () actual)
|
(check-equal?: (call-with-values (ann (λ () actual)
|
||||||
(-> #,(if (attribute type)
|
(-> #,(if (attribute type)
|
||||||
#'type
|
#'type.expanded
|
||||||
#'AnyValues)))
|
#'AnyValues)))
|
||||||
(λ l l))
|
(λ l l))
|
||||||
(list expected ...)))]))
|
(list expected ...)))]))
|
||||||
|
|
21
test/test-traversal-2.rkt
Normal file
21
test/test-traversal-2.rkt
Normal file
|
@ -0,0 +1,21 @@
|
||||||
|
#lang typed/racket
|
||||||
|
|
||||||
|
(require "../traversal.hl.rkt"
|
||||||
|
type-expander
|
||||||
|
phc-adt
|
||||||
|
"ck.rkt")
|
||||||
|
(adt-init)
|
||||||
|
|
||||||
|
#;(define-type Foo (Listof String))
|
||||||
|
|
||||||
|
(define-fold f₁ t₁ (tagged tg [a String] [b Boolean]) String)
|
||||||
|
|
||||||
|
(define (string->symbol+acc [x : String] [acc : Integer])
|
||||||
|
(values (string->symbol x) (add1 acc)))
|
||||||
|
|
||||||
|
(check-equal?-values:
|
||||||
|
((f₁ string? string->symbol+acc) (tagged tg [a "abc"] [b #f]) 0)
|
||||||
|
: (Values (tagged tg [a Symbol] [b Boolean]) Integer)
|
||||||
|
(tagged tg [a 'abc] [b #f]) 1)
|
||||||
|
|
||||||
|
|
124
traversal.hl.rkt
124
traversal.hl.rkt
|
@ -145,8 +145,10 @@ way up, so that a simple identity function can be applied in these cases.
|
||||||
|
|
||||||
@chunk[<define-fold-prepare>
|
@chunk[<define-fold-prepare>
|
||||||
(type-cases
|
(type-cases
|
||||||
(whole-type => _the-type _the-code the-defs …)
|
(whole-type #:to _the-type
|
||||||
#:literals (Null Pairof Listof List Vectorof Vector)
|
#:using _the-code
|
||||||
|
#:with-defintitions the-defs …)
|
||||||
|
#:literals (Null Pairof Listof List Vectorof Vector U tagged)
|
||||||
<type-cases>)]
|
<type-cases>)]
|
||||||
|
|
||||||
@chunk[<type-cases>
|
@chunk[<type-cases>
|
||||||
|
@ -155,49 +157,120 @@ way up, so that a simple identity function can be applied in these cases.
|
||||||
(syntax->list #'([type-to-replaceᵢ updateᵢ _Tᵢ] …)))
|
(syntax->list #'([type-to-replaceᵢ updateᵢ _Tᵢ] …)))
|
||||||
#:when (attribute info)
|
#:when (attribute info)
|
||||||
#:with (_ update T) #'info
|
#:with (_ update T) #'info
|
||||||
=> T
|
|
||||||
|
#:to
|
||||||
|
T
|
||||||
|
|
||||||
|
#:using
|
||||||
(update v acc)]]
|
(update v acc)]]
|
||||||
|
|
||||||
@chunk[<type-cases>
|
@chunk[<type-cases>
|
||||||
[(~or Null (List))
|
[(~or Null (List))
|
||||||
=> Null
|
|
||||||
|
#:to
|
||||||
|
Null
|
||||||
|
|
||||||
|
#:using
|
||||||
(values v acc)]]
|
(values v acc)]]
|
||||||
|
|
||||||
@chunk[<type-cases>
|
@chunk[<type-cases>
|
||||||
[(Pairof X Y)
|
[(Pairof X Y)
|
||||||
=> (Pairof (tx _Tᵢ …) (ty _Tᵢ …))
|
|
||||||
|
#:to
|
||||||
|
(Pairof (tx _Tᵢ …) (ty _Tᵢ …))
|
||||||
|
|
||||||
|
#:using
|
||||||
(let*-values ([(result-x acc-x) ((fx . _args) (car v) acc)]
|
(let*-values ([(result-x acc-x) ((fx . _args) (car v) acc)]
|
||||||
[(result-y acc-y) ((fy . _args) (cdr v) acc-x)])
|
[(result-y acc-y) ((fy . _args) (cdr v) acc-x)])
|
||||||
(values (cons result-x result-y) acc-y))
|
(values (cons result-x result-y) acc-y))
|
||||||
|
|
||||||
|
#:with-defintitions
|
||||||
(define-fold fx tx X type-to-replaceᵢ …)
|
(define-fold fx tx X type-to-replaceᵢ …)
|
||||||
(define-fold fy ty Y type-to-replaceᵢ …)]]
|
(define-fold fy ty Y type-to-replaceᵢ …)]]
|
||||||
|
|
||||||
@chunk[<type-cases>
|
@chunk[<type-cases>
|
||||||
[(Listof X)
|
[(Listof X)
|
||||||
=> (Listof (te _Tᵢ …))
|
|
||||||
|
#:to
|
||||||
|
(Listof (te _Tᵢ …))
|
||||||
|
|
||||||
|
#:using
|
||||||
(foldl-map (fe . _args) acc v)
|
(foldl-map (fe . _args) acc v)
|
||||||
|
|
||||||
|
#:with-defintitions
|
||||||
(define-fold fe te X type-to-replaceᵢ …)]]
|
(define-fold fe te X type-to-replaceᵢ …)]]
|
||||||
|
|
||||||
@chunk[<type-cases>
|
@chunk[<type-cases>
|
||||||
[(Vectorof X)
|
[(Vectorof X)
|
||||||
=> (Vectorof (te _Tᵢ …))
|
|
||||||
|
#:to
|
||||||
|
(Vectorof (te _Tᵢ …))
|
||||||
|
|
||||||
|
#:using
|
||||||
(vector->immutable-vector
|
(vector->immutable-vector
|
||||||
(list->vector
|
(list->vector
|
||||||
(foldl-map (fe . _args) acc (vector->list v))))
|
(foldl-map (fe . _args) acc (vector->list v))))
|
||||||
|
|
||||||
|
#:with-defintitions
|
||||||
(define-fold fe te X type-to-replaceᵢ …)]]
|
(define-fold fe te X type-to-replaceᵢ …)]]
|
||||||
|
|
||||||
@chunk[<type-cases>
|
@chunk[<type-cases>
|
||||||
[(List X Y ...)
|
[(List X Y …)
|
||||||
=> (Pairof (tx _Tᵢ …) (ty* _Tᵢ …))
|
|
||||||
|
#:to
|
||||||
|
(Pairof (tx _Tᵢ …) (ty* _Tᵢ …))
|
||||||
|
|
||||||
|
#:using
|
||||||
(let*-values ([(result-x acc-x) ((fx . _args) (car v) acc)]
|
(let*-values ([(result-x acc-x) ((fx . _args) (car v) acc)]
|
||||||
[(result-y* acc-y*) ((fy* . _args) (cdr v) acc-x)])
|
[(result-y* acc-y*) ((fy* . _args) (cdr v) acc-x)])
|
||||||
(values (cons result-x result-y*) acc-y*))
|
(values (cons result-x result-y*) acc-y*))
|
||||||
|
|
||||||
|
#:with-defintitions
|
||||||
(define-fold fx tx X type-to-replaceᵢ …)
|
(define-fold fx tx X type-to-replaceᵢ …)
|
||||||
(define-fold fy* ty* (List Y ...) type-to-replaceᵢ …)]]
|
(define-fold fy* ty* (List Y …) type-to-replaceᵢ …)]]
|
||||||
|
|
||||||
|
@chunk[<type-cases>
|
||||||
|
[(U X …)
|
||||||
|
|
||||||
|
#:to
|
||||||
|
(U (tx _Tᵢ …))
|
||||||
|
|
||||||
|
#:using
|
||||||
|
(dispatch-union ([type-to-replaceᵢ Aᵢ predicateᵢ]
|
||||||
|
…)
|
||||||
|
[X v ((fx . _args) v acc)]
|
||||||
|
…)
|
||||||
|
|
||||||
|
#:with-defintitions
|
||||||
|
(define-fold fx tx X type-to-replaceᵢ …)
|
||||||
|
…]]
|
||||||
|
|
||||||
|
@chunk[<type-cases>
|
||||||
|
[(tagged _name [_field (~optional :colon) _X] …
|
||||||
|
{~do (define-temp-ids "_fx" (_X …))}
|
||||||
|
{~do (define-temp-ids "_tx" (_X …))}
|
||||||
|
{~do (define-temp-ids "_result" (_X …))})
|
||||||
|
|
||||||
|
#:to
|
||||||
|
(tagged _name [_field : (_tx _Tᵢ …)] …)
|
||||||
|
|
||||||
|
#:using
|
||||||
|
(let*-values ([(_result acc) ((_fx . _args) (uniform-get v _field) acc)]
|
||||||
|
…)
|
||||||
|
(values (tagged _name [_field _result] …)
|
||||||
|
acc))
|
||||||
|
|
||||||
|
#:with-defintitions
|
||||||
|
(define-fold _fx _tx _X type-to-replaceᵢ …)
|
||||||
|
…]]
|
||||||
|
|
||||||
@chunk[<type-cases>
|
@chunk[<type-cases>
|
||||||
[else-T
|
[else-T
|
||||||
=> else-T
|
|
||||||
|
#:to
|
||||||
|
else-T
|
||||||
|
|
||||||
|
#:using
|
||||||
(values v acc)]]
|
(values v acc)]]
|
||||||
|
|
||||||
where @racket[foldl-map] is defined as:
|
where @racket[foldl-map] is defined as:
|
||||||
|
@ -219,17 +292,22 @@ where @racket[foldl-map] is defined as:
|
||||||
@chunk[<type-cases-macro>
|
@chunk[<type-cases-macro>
|
||||||
(define-syntax type-cases
|
(define-syntax type-cases
|
||||||
(syntax-parser
|
(syntax-parser
|
||||||
#:literals (=>)
|
[(_ (whole-type #:to the-type
|
||||||
[(_ (whole-type => the-type the-code the-defs (~literal …))
|
#:using the-code
|
||||||
|
#:with-defintitions the-defs (~literal …))
|
||||||
#:literals (lit …)
|
#:literals (lit …)
|
||||||
(Pat opts … => transform-type transform-code transform-defs …)
|
(Pat opts …
|
||||||
|
#:to transform-type
|
||||||
|
#:using transform-code
|
||||||
|
(~optional (~seq #:with-defintitions transform-defs …)
|
||||||
|
#:defaults ([(transform-defs 1) (list)])))
|
||||||
…)
|
…)
|
||||||
#'(define/with-syntax (the-type the-code the-defs (… …))
|
#'(define/with-syntax (the-type the-code the-defs (… …))
|
||||||
(syntax-parse #'whole-type
|
(syntax-parse #'whole-type
|
||||||
#:literals (lit …)
|
#:literals (lit …)
|
||||||
[Pat opts …
|
[Pat opts …
|
||||||
(template
|
(template
|
||||||
(transform-type transform-code transform-defs …))]
|
(transform-type transform-code transform-defs …))]
|
||||||
…))]))]
|
…))]))]
|
||||||
|
|
||||||
@chunk[<define-fold-result>
|
@chunk[<define-fold-result>
|
||||||
|
@ -238,13 +316,13 @@ where @racket[foldl-map] is defined as:
|
||||||
(define-type (_type-name _Tᵢ …) _the-type)
|
(define-type (_type-name _Tᵢ …) _the-type)
|
||||||
|
|
||||||
(: _function-name (∀ (_Aᵢ … _Bᵢ … Acc)
|
(: _function-name (∀ (_Aᵢ … _Bᵢ … Acc)
|
||||||
(→ (?@ (→ Any Boolean : _Aᵢ)
|
(→ (?@ (→ Any Boolean : _Aᵢ)
|
||||||
(→ _Aᵢ Acc (Values _Bᵢ Acc)))
|
(→ _Aᵢ Acc (Values _Bᵢ Acc)))
|
||||||
…
|
…
|
||||||
(→ (_type-name _Aᵢ …)
|
(→ (_type-name _Aᵢ …)
|
||||||
Acc
|
Acc
|
||||||
(Values (_type-name _Bᵢ …)
|
(Values (_type-name _Bᵢ …)
|
||||||
Acc)))))
|
Acc)))))
|
||||||
(define ((_function-name . _args) v acc)
|
(define ((_function-name . _args) v acc)
|
||||||
_the-code)]
|
_the-code)]
|
||||||
|
|
||||||
|
@ -252,6 +330,8 @@ where @racket[foldl-map] is defined as:
|
||||||
|
|
||||||
@chunk[<*>
|
@chunk[<*>
|
||||||
(require phc-toolkit
|
(require phc-toolkit
|
||||||
|
type-expander
|
||||||
|
phc-adt
|
||||||
(for-syntax racket/base
|
(for-syntax racket/base
|
||||||
phc-toolkit/untyped
|
phc-toolkit/untyped
|
||||||
racket/syntax
|
racket/syntax
|
||||||
|
|
Loading…
Reference in New Issue
Block a user