checkpoint
This commit is contained in:
parent
02e4e94fb1
commit
9993a038f2
|
@ -1,6 +1,8 @@
|
||||||
TODO add ; back into braces but don't have , because of conflict with
|
TODO add ; back into braces but don't have , because of conflict with
|
||||||
its ,
|
its ,
|
||||||
|
|
||||||
|
TODO think about interface language part of theory/interface/modules
|
||||||
|
|
||||||
TODO add syntax property for def transformer on RHS (for function call
|
TODO add syntax property for def transformer on RHS (for function call
|
||||||
results, alloc, etc)
|
results, alloc, etc)
|
||||||
|
|
||||||
|
|
|
@ -478,6 +478,8 @@
|
||||||
|
|
||||||
;; Interfaces & Classes
|
;; Interfaces & Classes
|
||||||
|
|
||||||
|
(struct object (interface->implementation rep))
|
||||||
|
|
||||||
(define-syntax interface
|
(define-syntax interface
|
||||||
(singleton-struct
|
(singleton-struct
|
||||||
#:property prop:procedure
|
#:property prop:procedure
|
||||||
|
@ -485,21 +487,33 @@
|
||||||
(raise-syntax-error 'interface "Illegal outside def" stx))
|
(raise-syntax-error 'interface "Illegal outside def" stx))
|
||||||
#:methods remix:gen:def-transformer
|
#:methods remix:gen:def-transformer
|
||||||
[(define (def-transform _ stx)
|
[(define (def-transform _ stx)
|
||||||
;; XXX
|
(syntax-parse stx
|
||||||
#'(void))]))
|
#:literals (remix:#%brackets remix:def interface)
|
||||||
|
;; XXX support parameters?
|
||||||
|
[(remix:def (remix:#%brackets interface int:id)
|
||||||
|
;; XXX support properties?
|
||||||
|
;; XXX make expandable position
|
||||||
|
v:id ...)
|
||||||
|
(syntax/loc stx
|
||||||
|
;; XXX instead, make an int-vtable and then a separate int
|
||||||
|
;; def transformer that looks at objects.
|
||||||
|
(remix:def (remix:#%brackets theory int)
|
||||||
|
;; XXX add a property for interfaces
|
||||||
|
;; XXX support defaults?
|
||||||
|
v ...))]))]))
|
||||||
|
|
||||||
(define-syntax-parameter clayout
|
(define-syntax-parameter representation
|
||||||
(λ (stx)
|
(λ (stx)
|
||||||
(raise-syntax-error 'clayout "Illegal outside class" stx)))
|
(raise-syntax-error 'representation "Illegal outside class" stx)))
|
||||||
(define-syntax-parameter new
|
(define-syntax-parameter new
|
||||||
(λ (stx)
|
(λ (stx)
|
||||||
(raise-syntax-error 'new "Illegal outside class" stx)))
|
(raise-syntax-error 'new "Illegal outside class" stx)))
|
||||||
(define-syntax-parameter this
|
(define-syntax-parameter this
|
||||||
(λ (stx)
|
(λ (stx)
|
||||||
(raise-syntax-error 'this "Illegal outside class" stx)))
|
(raise-syntax-error 'this "Illegal outside class" stx)))
|
||||||
(define-syntax-parameter implements
|
(define-syntax-parameter implementation
|
||||||
(λ (stx)
|
(λ (stx)
|
||||||
(raise-syntax-error 'implements "Illegal outside class" stx)))
|
(raise-syntax-error 'implementation "Illegal outside class" stx)))
|
||||||
|
|
||||||
(define-syntax class
|
(define-syntax class
|
||||||
(singleton-struct
|
(singleton-struct
|
||||||
|
@ -508,15 +522,17 @@
|
||||||
(raise-syntax-error 'class "Illegal outside def" stx))
|
(raise-syntax-error 'class "Illegal outside def" stx))
|
||||||
#:methods remix:gen:def-transformer
|
#:methods remix:gen:def-transformer
|
||||||
[(define (def-transform _ stx)
|
[(define (def-transform _ stx)
|
||||||
|
;; XXX ensure everything is expandable
|
||||||
;; XXX
|
;; XXX
|
||||||
#'(void))]))
|
#'(void))]))
|
||||||
|
|
||||||
(provide interface
|
(provide interface
|
||||||
clayout
|
representation
|
||||||
|
(rename-out [representation rep])
|
||||||
new
|
new
|
||||||
this
|
this
|
||||||
implements
|
implementation
|
||||||
(rename-out implements impl)
|
(rename-out [implementation impl])
|
||||||
class)
|
class)
|
||||||
|
|
||||||
;; xxx data (fixed set of interfaces)
|
;; xxx data (fixed set of interfaces)
|
||||||
|
|
|
@ -521,6 +521,7 @@ def x4
|
||||||
;; vector-ref) and the operation couldn't be inlined. (Although if
|
;; vector-ref) and the operation couldn't be inlined. (Although if
|
||||||
;; the generic function were inlined, then it could, presumably.)
|
;; the generic function were inlined, then it could, presumably.)
|
||||||
(def (monoid-id-test [Monoid m] a)
|
(def (monoid-id-test [Monoid m] a)
|
||||||
|
;; Notice the syntax `m.(op x y)` as short-hand for `((m.op) x y)`
|
||||||
{((m.op) a m.id) ≡ m.(op m.id a)}))
|
{((m.op) a m.id) ≡ m.(op m.id a)}))
|
||||||
|
|
||||||
;; A model is an object that satisfies the theory
|
;; A model is an object that satisfies the theory
|
||||||
|
@ -550,15 +551,16 @@ def x4
|
||||||
;; xxx make a macro for "interface of layout's fields"
|
;; xxx make a macro for "interface of layout's fields"
|
||||||
c r)
|
c r)
|
||||||
|
|
||||||
|
;; A class is a representation, a constructor, and implementations of
|
||||||
|
;; interfaces.
|
||||||
(def [class Circle]
|
(def [class Circle]
|
||||||
(def [clayout]
|
(def [rep] circle) ;; rep = representation
|
||||||
circle)
|
|
||||||
(def ([new] x y r)
|
(def ([new] x y r)
|
||||||
(this.#:alloc [c (posn.#:alloc [x x] [y y])]
|
(this.#:alloc [c (posn.#:alloc [x x] [y y])]
|
||||||
[r r]))
|
[r r]))
|
||||||
|
|
||||||
;; xxx make a macro from "layout's fields implements this interface"
|
;; xxx make a macro from "layout's fields implements this interface"
|
||||||
(def [implements Circle<%>]
|
(def [implementation Circle<%>]
|
||||||
[(c) this.c]
|
[(c) this.c]
|
||||||
[(r) this.r])
|
[(r) this.r])
|
||||||
|
|
||||||
|
@ -570,11 +572,11 @@ def x4
|
||||||
[(area)
|
[(area)
|
||||||
{3 * this.r * this.r}]))
|
{3 * this.r * this.r}]))
|
||||||
|
|
||||||
;; XXX allow w/o #:new?
|
;; XXX allow w/o #:new?, like layout
|
||||||
;; XXX
|
|
||||||
#;#;
|
|
||||||
(def [Circle C1] (Circle.#:new 1 2 3))
|
(def [Circle C1] (Circle.#:new 1 2 3))
|
||||||
(module+ test
|
(module+ test
|
||||||
|
;; If you know something is a particular class, then you can access
|
||||||
|
;; its implementations directly. This is more efficient.
|
||||||
{C1.Circle<%>.c.x ≡ 1}
|
{C1.Circle<%>.c.x ≡ 1}
|
||||||
{C1.Circle<%>.c.y ≡ 2}
|
{C1.Circle<%>.c.y ≡ 2}
|
||||||
{C1.Circle<%>.r ≡ 3}
|
{C1.Circle<%>.r ≡ 3}
|
||||||
|
@ -582,6 +584,22 @@ def x4
|
||||||
(def [Circle C1′] (C1.2d<%>.translate 3 2))
|
(def [Circle C1′] (C1.2d<%>.translate 3 2))
|
||||||
{C1′.Circle<%>.c.x ≡ 4}
|
{C1′.Circle<%>.c.x ≡ 4}
|
||||||
{C1′.Circle<%>.c.y ≡ 4}
|
{C1′.Circle<%>.c.y ≡ 4}
|
||||||
{C1′.Circle<%>.r ≡ 3})
|
{C1′.Circle<%>.r ≡ 3}
|
||||||
|
;; In contrast, when you access them as their interfaces, a lookup
|
||||||
|
;; is done.
|
||||||
|
(def [2d<%> C1-as-2d] C1)
|
||||||
|
{C1-as-2d.(area) ≡ 27}
|
||||||
|
(def [Circle<%> C1-as-Circ] C1)
|
||||||
|
{C1-as-Circ.c.x ≡ 1}
|
||||||
|
{C1-as-Circ.c.y ≡ 2}
|
||||||
|
{C1-as-Circ.r ≡ 3})
|
||||||
|
|
||||||
|
(module+ test
|
||||||
|
;; Like theories, you can define functions that are generic over an
|
||||||
|
;; interface.
|
||||||
|
(def (squarea [2d<%> o])
|
||||||
|
{o.(area) * o.(area)})
|
||||||
|
{(squarea C1) ≡ 729}
|
||||||
|
;; The default behavior of class dot-transformers on unknown methods
|
||||||
|
;; is to treat it as a generic function.
|
||||||
|
{C1.(squarea) ≡ 729})
|
||||||
|
|
Loading…
Reference in New Issue
Block a user