infix
This commit is contained in:
parent
619fbb9217
commit
74c00f4962
|
@ -4,7 +4,7 @@ DONE demonstrate how @ everywhere enables non-() macros like datalog
|
||||||
|
|
||||||
DONE {} particular => (#%braces)
|
DONE {} particular => (#%braces)
|
||||||
|
|
||||||
TODO {} infix
|
DONE {} infix
|
||||||
|
|
||||||
DONE [] particular => (#%brackets)
|
DONE [] particular => (#%brackets)
|
||||||
|
|
||||||
|
@ -60,7 +60,7 @@ TODO No set! (use data-structure mutation and :=)
|
||||||
|
|
||||||
TODO No effects or expressions at top-level (controversial, mf says wrong) [set-once!]
|
TODO No effects or expressions at top-level (controversial, mf says wrong) [set-once!]
|
||||||
|
|
||||||
TODO "Versioned" libraries
|
DONE "Versioned" libraries
|
||||||
|
|
||||||
TODO Make macros less weird and more like programs, by...
|
TODO Make macros less weird and more like programs, by...
|
||||||
|
|
||||||
|
|
|
@ -1,3 +1,4 @@
|
||||||
#lang racket/base
|
#lang racket/base
|
||||||
;; xxx fill out
|
;; xxx fill out
|
||||||
(provide + < /)
|
(provide + < / * -
|
||||||
|
bitwise-and)
|
||||||
|
|
|
@ -1,5 +1,8 @@
|
||||||
#lang racket/base
|
#lang racket/base
|
||||||
(require (for-syntax racket/base
|
(require (for-syntax racket/base
|
||||||
|
racket/list
|
||||||
|
racket/match
|
||||||
|
racket/generic
|
||||||
syntax/parse))
|
syntax/parse))
|
||||||
|
|
||||||
(define-syntax (def stx)
|
(define-syntax (def stx)
|
||||||
|
@ -32,8 +35,79 @@
|
||||||
(syntax/loc stx
|
(syntax/loc stx
|
||||||
(remix-block . body))]))
|
(remix-block . body))]))
|
||||||
|
|
||||||
|
(begin-for-syntax
|
||||||
|
(define-generics binary-operator
|
||||||
|
[binary-operator-precedence binary-operator])
|
||||||
|
(define (operator-chars? s)
|
||||||
|
(not
|
||||||
|
(ormap (λ (c) (or (char-alphabetic? c)
|
||||||
|
(char-numeric? c)))
|
||||||
|
(string->list s))))
|
||||||
|
(define-syntax-class operator-sym
|
||||||
|
(pattern op:identifier
|
||||||
|
#:when (operator-chars? (symbol->string (syntax->datum #'op)))))
|
||||||
|
(define PRECEDENCE-TABLE
|
||||||
|
(hasheq '* 30 '/ 30
|
||||||
|
'+ 40 '- 40
|
||||||
|
'< 60 '<= 60
|
||||||
|
'> 60 '>= 60
|
||||||
|
'= 70))
|
||||||
|
(define (shunting-yard:precendence op)
|
||||||
|
(define v (syntax-local-value op (λ () #f)))
|
||||||
|
(or (and v (binary-operator? v) (binary-operator-precedence v))
|
||||||
|
(hash-ref PRECEDENCE-TABLE (syntax->datum op) 150)))
|
||||||
|
|
||||||
|
(define (shunting-yard:consume-input input output operators)
|
||||||
|
(match input
|
||||||
|
['()
|
||||||
|
(shunting-yard:pop-operators output operators)]
|
||||||
|
[(cons token input)
|
||||||
|
(syntax-parse token
|
||||||
|
#:literals (unquote)
|
||||||
|
[(~or (unquote (~and op1:expr (~not _:operator-sym))) op1:operator-sym)
|
||||||
|
(define-values (output-p operators-p)
|
||||||
|
(shunting-yard:push-operator output operators #'op1))
|
||||||
|
(shunting-yard:consume-input input output-p operators-p)]
|
||||||
|
[(~or (unquote arg:operator-sym) arg:expr)
|
||||||
|
(shunting-yard:consume-input input (cons #'arg output) operators)])]))
|
||||||
|
(define (shunting-yard:push-operator output operators op1)
|
||||||
|
(match operators
|
||||||
|
['()
|
||||||
|
(values output (cons op1 operators))]
|
||||||
|
[(cons op2 operators-p)
|
||||||
|
(cond
|
||||||
|
[(<= (shunting-yard:precendence op2) (shunting-yard:precendence op1))
|
||||||
|
(shunting-yard:push-operator
|
||||||
|
(shunting-yard:push-operator-to-output op2 output)
|
||||||
|
operators-p op1)]
|
||||||
|
[else
|
||||||
|
(values output (cons op1 operators))])]))
|
||||||
|
(define (shunting-yard:pop-operators output operators)
|
||||||
|
(match operators
|
||||||
|
['()
|
||||||
|
(match output
|
||||||
|
[(list result)
|
||||||
|
result]
|
||||||
|
[_
|
||||||
|
(error 'shunting-yard:pop-operators "Too much output: ~v" output)])]
|
||||||
|
[(cons op operators)
|
||||||
|
(shunting-yard:pop-operators
|
||||||
|
(shunting-yard:push-operator-to-output op output)
|
||||||
|
operators)]))
|
||||||
|
(define (shunting-yard:push-operator-to-output op output)
|
||||||
|
(syntax-parse output
|
||||||
|
[(arg2:expr arg1:expr output:expr ...)
|
||||||
|
(cons (quasisyntax/loc op
|
||||||
|
(#,op arg1 arg2))
|
||||||
|
(syntax->list
|
||||||
|
#'(output ...)))])))
|
||||||
(define-syntax (#%braces stx)
|
(define-syntax (#%braces stx)
|
||||||
(syntax-parse stx))
|
(syntax-parse stx
|
||||||
|
[(_ input-tokens:expr ...)
|
||||||
|
(shunting-yard:consume-input
|
||||||
|
(syntax->list #'(input-tokens ...))
|
||||||
|
empty
|
||||||
|
empty)]))
|
||||||
|
|
||||||
(define-syntax (#%dot stx)
|
(define-syntax (#%dot stx)
|
||||||
(syntax-parse stx))
|
(syntax-parse stx))
|
||||||
|
@ -59,9 +133,13 @@
|
||||||
[remix-cond cond])
|
[remix-cond cond])
|
||||||
#%brackets
|
#%brackets
|
||||||
#%braces
|
#%braces
|
||||||
|
(for-syntax gen:binary-operator
|
||||||
|
binary-operator?
|
||||||
|
binary-operator-precedence)
|
||||||
#%dot
|
#%dot
|
||||||
#%app
|
#%app
|
||||||
#%datum
|
#%datum
|
||||||
|
unquote
|
||||||
module
|
module
|
||||||
module*
|
module*
|
||||||
module+)
|
module+)
|
||||||
|
|
|
@ -34,4 +34,35 @@
|
||||||
path(X, Y)?
|
path(X, Y)?
|
||||||
}
|
}
|
||||||
|
|
||||||
|
(def v7
|
||||||
|
{3 + 4})
|
||||||
|
(module+ test
|
||||||
|
v7)
|
||||||
|
|
||||||
|
(def v-26
|
||||||
|
{2 * 3 - 48 / 4 - 4 * 5})
|
||||||
|
(module+ test
|
||||||
|
v-26)
|
||||||
|
|
||||||
|
(def v15
|
||||||
|
{v7 * 2 + 1})
|
||||||
|
(module+ test
|
||||||
|
v15)
|
||||||
|
|
||||||
|
(def v14
|
||||||
|
(def (f x y) (+ x y))
|
||||||
|
{v7 ,f v7})
|
||||||
|
(module+ test
|
||||||
|
v14)
|
||||||
|
|
||||||
|
(def v1
|
||||||
|
(def & bitwise-and)
|
||||||
|
{5 & 1})
|
||||||
|
(module+ test
|
||||||
|
v1)
|
||||||
|
|
||||||
|
(def v9
|
||||||
|
(def & 2)
|
||||||
|
{v7 + ,&})
|
||||||
|
(module+ test
|
||||||
|
v9)
|
||||||
|
|
Loading…
Reference in New Issue
Block a user