adding parents
This commit is contained in:
parent
1e82bd68e9
commit
39041fae09
|
@ -1,11 +1,13 @@
|
||||||
#lang racket/base
|
#lang racket/base
|
||||||
(require (for-syntax
|
(require (for-syntax
|
||||||
racket/base
|
racket/base
|
||||||
|
syntax/quote
|
||||||
syntax/parse
|
syntax/parse
|
||||||
racket/syntax
|
racket/syntax
|
||||||
racket/generic
|
racket/generic
|
||||||
racket/format
|
racket/format
|
||||||
racket/list
|
racket/list
|
||||||
|
racket/match
|
||||||
(prefix-in remix: remix/stx0)
|
(prefix-in remix: remix/stx0)
|
||||||
remix/stx/singleton-struct0
|
remix/stx/singleton-struct0
|
||||||
(for-syntax racket/base
|
(for-syntax racket/base
|
||||||
|
@ -185,7 +187,11 @@
|
||||||
|
|
||||||
(begin-for-syntax
|
(begin-for-syntax
|
||||||
;; XXX fill this in for parents, etc
|
;; XXX fill this in for parents, etc
|
||||||
(define-generics layout)
|
(define-generics layout
|
||||||
|
(layout-planner layout)
|
||||||
|
;; xxx the accessors seem to not be around anyways, so instead,
|
||||||
|
;; this should just be a mapping produced by the planner.
|
||||||
|
(layout-field->acc layout))
|
||||||
(begin-for-syntax
|
(begin-for-syntax
|
||||||
(define-generics layout-planner
|
(define-generics layout-planner
|
||||||
(layout-planner-mutable? layout-planner)))
|
(layout-planner-mutable? layout-planner)))
|
||||||
|
@ -208,36 +214,62 @@
|
||||||
(syntax-parse stx
|
(syntax-parse stx
|
||||||
#:literals (remix:#%brackets)
|
#:literals (remix:#%brackets)
|
||||||
[(def (remix:#%brackets me:id name:id)
|
[(def (remix:#%brackets me:id name:id)
|
||||||
|
(~optional (~and (~seq #:parent (~var parent (static layout? "layout")))
|
||||||
|
(~bind [parent-va (attribute parent.value)]))
|
||||||
|
#:defaults ([parent-va #f]))
|
||||||
f:id ...)
|
f:id ...)
|
||||||
;; xxx check for duplicates in f
|
(define parent-v (attribute parent-va))
|
||||||
|
(define the-planner
|
||||||
|
(or (and parent-v (layout-planner parent-v))
|
||||||
|
;; xxx allow the default planner to be customized and
|
||||||
|
;; ensure it is equal to the parent's
|
||||||
|
#'layout-immutable))
|
||||||
|
(define parent-f->acc
|
||||||
|
(or (and parent-v (layout-field->acc parent-v))
|
||||||
|
(hasheq)))
|
||||||
|
(define f->acc
|
||||||
|
(for/fold ([base parent-f->acc])
|
||||||
|
([the-f (in-list (syntax->datum #'(f ...)))]
|
||||||
|
[the-idx (in-naturals (hash-count parent-f->acc))])
|
||||||
|
(when (hash-has-key? base the-f)
|
||||||
|
(raise-syntax-error 'layout
|
||||||
|
(format "duplicate field ~a in layout"
|
||||||
|
the-f)
|
||||||
|
stx
|
||||||
|
the-f))
|
||||||
|
(define the-name-f (format-id #f "~a-~a" #'name the-f))
|
||||||
|
(hash-set base the-f (cons the-name-f the-idx))))
|
||||||
(with-syntax* ([name-alloc (format-id #f "~a-alloc" #'name)]
|
(with-syntax* ([name-alloc (format-id #f "~a-alloc" #'name)]
|
||||||
[name-set (format-id #f "~a-set" #'name)]
|
[name-set (format-id #f "~a-set" #'name)]
|
||||||
[(f-idx ...)
|
[((all-f all-name-f all-f-idx) ...)
|
||||||
(for/list ([i (in-naturals)]
|
(for/list ([(the-f v) (in-hash f->acc)])
|
||||||
[f (in-list (syntax->list #'(f ...)))])
|
(match-define (cons the-name-f the-f-idx) v)
|
||||||
i)]
|
(list the-f the-name-f the-f-idx))]
|
||||||
[(name-f ...)
|
[stx-the-planner the-planner]
|
||||||
(for/list ([f (in-list (syntax->list #'(f ...)))])
|
[stx-f->acc f->acc])
|
||||||
(format-id #f "~a-~a" #'name f))])
|
|
||||||
(syntax/loc stx
|
(syntax/loc stx
|
||||||
(begin
|
(begin
|
||||||
(begin-for-syntax
|
(begin-for-syntax
|
||||||
(define f->idx*acc
|
(define f->acc stx-f->acc)
|
||||||
;; xxx base on parent's
|
|
||||||
(make-immutable-hasheq
|
|
||||||
(list (cons 'f (cons f-idx #'name-f))
|
|
||||||
...)))
|
|
||||||
(define available-fields
|
(define available-fields
|
||||||
(sort (hash-keys f->idx*acc)
|
(sort (hash-keys f->acc)
|
||||||
string<=?
|
string<=?
|
||||||
#:key symbol->string))
|
#:key symbol->string))
|
||||||
|
(define ordered-fields
|
||||||
|
(sort (hash-keys f->acc)
|
||||||
|
<=
|
||||||
|
#:key (λ (x)
|
||||||
|
(cdr (hash-ref f->acc x)))))
|
||||||
(define-syntax-class name-arg
|
(define-syntax-class name-arg
|
||||||
#:attributes (lhs rhs)
|
#:attributes (lhs rhs)
|
||||||
#:literals (remix:#%brackets)
|
#:literals (remix:#%brackets)
|
||||||
(pattern (remix:#%brackets lhs:id rhs:expr)
|
(pattern (remix:#%brackets lhs:id rhs:expr)
|
||||||
|
#:do [(define lhs-v (syntax->datum #'lhs))]
|
||||||
#:fail-unless
|
#:fail-unless
|
||||||
(hash-has-key? f->idx*acc (syntax->datum #'lhs))
|
(hash-has-key? f->acc lhs-v)
|
||||||
(format "valid field: ~a" available-fields)))
|
(format "invalid field given: ~a, valid fields are: ~a"
|
||||||
|
lhs-v
|
||||||
|
available-fields)))
|
||||||
(define-syntax-class name-args
|
(define-syntax-class name-args
|
||||||
#:attributes (f->rhs)
|
#:attributes (f->rhs)
|
||||||
(pattern (a:name-arg (... ...))
|
(pattern (a:name-arg (... ...))
|
||||||
|
@ -254,7 +286,7 @@
|
||||||
(syntax-parse stx
|
(syntax-parse stx
|
||||||
[(_ . args:name-args)
|
[(_ . args:name-args)
|
||||||
(with-syntax ([(f-val (... ...))
|
(with-syntax ([(f-val (... ...))
|
||||||
(for/list ([this-f (in-list '(f ...))])
|
(for/list ([this-f (in-list ordered-fields)])
|
||||||
(hash-ref (attribute args.f->rhs)
|
(hash-ref (attribute args.f->rhs)
|
||||||
this-f
|
this-f
|
||||||
(λ ()
|
(λ ()
|
||||||
|
@ -271,8 +303,8 @@
|
||||||
[(_ base:expr . args:name-args)
|
[(_ base:expr . args:name-args)
|
||||||
(with-syntax* ([base-id (generate-temporary #'base)]
|
(with-syntax* ([base-id (generate-temporary #'base)]
|
||||||
[(f-val (... ...))
|
[(f-val (... ...))
|
||||||
(for/list ([this-f (in-list '(f ...))]
|
(for/list ([this-f (in-list ordered-fields)])
|
||||||
[this-name-f (in-list '(name-f ...))])
|
(define this-name-f (car (hash-ref f->acc this-f)))
|
||||||
(hash-ref (attribute args.f->rhs)
|
(hash-ref (attribute args.f->rhs)
|
||||||
this-f
|
this-f
|
||||||
(λ ()
|
(λ ()
|
||||||
|
@ -284,7 +316,7 @@
|
||||||
(vector-immutable f-val (... ...)))))]))
|
(vector-immutable f-val (... ...)))))]))
|
||||||
(begin-encourage-inline
|
(begin-encourage-inline
|
||||||
;; xxx push this in representation planner
|
;; xxx push this in representation planner
|
||||||
(define (name-f v) (unsafe-vector*-ref v f-idx))
|
(define (all-name-f v) (unsafe-vector*-ref v all-f-idx))
|
||||||
...)
|
...)
|
||||||
(define-syntax name
|
(define-syntax name
|
||||||
(phase1:static-interface
|
(phase1:static-interface
|
||||||
|
@ -292,11 +324,14 @@
|
||||||
(remix:#%brackets #:set name-set)
|
(remix:#%brackets #:set name-set)
|
||||||
(remix:#%brackets #:= name-set)
|
(remix:#%brackets #:= name-set)
|
||||||
;; xxx add set! if planner says so
|
;; xxx add set! if planner says so
|
||||||
(remix:#%brackets f name-f)
|
(remix:#%brackets all-f all-name-f)
|
||||||
...
|
...
|
||||||
#:extensions
|
#:extensions
|
||||||
#:methods gen:layout
|
#:methods gen:layout
|
||||||
[])))))]))]))
|
[(define (layout-planner _)
|
||||||
|
#'stx-the-planner)
|
||||||
|
(define (layout-field->acc _)
|
||||||
|
f->acc)])))))]))]))
|
||||||
|
|
||||||
(provide (rename-out [phase0:layout layout])
|
(provide (rename-out [phase0:layout layout])
|
||||||
(for-meta 2
|
(for-meta 2
|
||||||
|
|
|
@ -347,16 +347,27 @@
|
||||||
{p3.x ≡ 8}
|
{p3.x ≡ 8}
|
||||||
{p3.y ≡ 7})
|
{p3.y ≡ 7})
|
||||||
|
|
||||||
;; XXX This is where I am
|
|
||||||
|
|
||||||
;; A layout can have a parent, which provides the guarantee that the
|
;; A layout can have a parent, which provides the guarantee that the
|
||||||
;; parent's functions will work on the child---meaning that whatever
|
;; parent's functions will work on the child---meaning that whatever
|
||||||
;; the layout ends up being (and you can't decide that), the two will
|
;; the layout ends up being (and you can't decide that), the two will
|
||||||
;; overlap in this specific way. A layout has one or zero parents.
|
;; overlap in this specific way. A layout has one or zero parents.
|
||||||
#;
|
|
||||||
(def [layout quat]
|
(def [layout quat]
|
||||||
#:parent posn
|
#:parent posn
|
||||||
z)
|
z)
|
||||||
|
(module+ test
|
||||||
|
(def [quat q1] (quat.#:alloc [x 1] [y 2] [z 3]))
|
||||||
|
{q1.x ≡ 1}
|
||||||
|
{q1.y ≡ 2}
|
||||||
|
{q1.z ≡ 3}
|
||||||
|
(def [posn qp1] q1)
|
||||||
|
{qp1.x ≡ 1}
|
||||||
|
{qp1.y ≡ 2}
|
||||||
|
(def [quat qpq1] qp1)
|
||||||
|
{qpq1.x ≡ 1}
|
||||||
|
{qpq1.y ≡ 2}
|
||||||
|
{qpq1.z ≡ 3})
|
||||||
|
|
||||||
|
;; XXX This is where I am
|
||||||
|
|
||||||
;; A layout's fields may be specified as other layouts. When the first
|
;; A layout's fields may be specified as other layouts. When the first
|
||||||
;; field is a layout, this is not necessarily the same thing as a
|
;; field is a layout, this is not necessarily the same thing as a
|
||||||
|
|
Loading…
Reference in New Issue
Block a user