-
Notifications
You must be signed in to change notification settings - Fork 76
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
coalton:match to codegen cl:case when possible #1399
base: main
Are you sure you want to change the base?
Changes from all commits
cb0c1d3
c513646
5b68c48
ddc7802
ac769e1
fed5068
eda1af1
cfeccf4
e0d4aa1
c54cc21
e27aa5d
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change | ||||||||
---|---|---|---|---|---|---|---|---|---|---|
|
@@ -172,6 +172,63 @@ | |||||||||
,(codegen-expression (match-branch-body (first (node-match-branches expr))) env) | ||||||||||
,(codegen-expression (match-branch-body (second (node-match-branches expr))) env)))) | ||||||||||
|
||||||||||
;; Do cl:if when the order is reversed as well (this method is getting fat) | ||||||||||
(when (and (equalp (node-type (node-match-expr expr)) tc:*boolean-type*) | ||||||||||
(= 2 (length (node-match-branches expr))) | ||||||||||
(equalp (match-branch-pattern (first (node-match-branches expr))) | ||||||||||
(make-pattern-constructor :type tc:*boolean-type* :name 'coalton:False :patterns nil)) | ||||||||||
(equalp (match-branch-pattern (second (node-match-branches expr))) | ||||||||||
(make-pattern-constructor :type tc:*boolean-type* :name 'coalton:True :patterns nil))) | ||||||||||
(return-from codegen-expression | ||||||||||
`(if ,(codegen-expression (node-match-expr expr) env) | ||||||||||
,(codegen-expression (match-branch-body (second (node-match-branches expr))) env) | ||||||||||
,(codegen-expression (match-branch-body (first (node-match-branches expr))) env)))) | ||||||||||
|
||||||||||
;; If we can use case, do that because a jump table is faster than cond | ||||||||||
(let ((ty (node-type (node-match-expr expr)))) | ||||||||||
(when (or | ||||||||||
;; Matching a number | ||||||||||
(find ty (tc:number-types) :test #'equalp) | ||||||||||
;; Matching a Char | ||||||||||
(equalp ty tc:*char-type*) | ||||||||||
;; Matching an Enum | ||||||||||
(and (tc:tycon-p ty) (tc:type-entry-enum-repr (tc:lookup-type env (tc:tycon-name ty))))) | ||||||||||
(return-from codegen-expression | ||||||||||
(let ((subexpr (codegen-expression (node-match-expr expr) env)) | ||||||||||
(match-var (gensym "MATCH"))) | ||||||||||
`(let ((,match-var | ||||||||||
,(if settings:*emit-type-annotations* | ||||||||||
`(the ,(tc:lisp-type (node-type (node-match-expr expr)) env) ,subexpr) | ||||||||||
subexpr))) | ||||||||||
(declare (ignorable ,match-var)) | ||||||||||
(case ,(codegen-expression (node-match-expr expr) env) | ||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This can be
Suggested change
or even
Suggested change
|
||||||||||
,@(loop :for branch :in (node-match-branches expr) | ||||||||||
:for pattern := (match-branch-pattern branch) | ||||||||||
:for expr := (codegen-expression (match-branch-body branch) env) | ||||||||||
:collect | ||||||||||
(multiple-value-bind (pred bindings) | ||||||||||
(codegen-pattern pattern match-var env) | ||||||||||
(declare (ignore pred)) | ||||||||||
(cond ((pattern-literal-p pattern) | ||||||||||
`(,(pattern-literal-value pattern) | ||||||||||
,expr)) | ||||||||||
((pattern-constructor-p pattern) | ||||||||||
(let* ((name (pattern-constructor-name pattern)) | ||||||||||
(entry (tc:lookup-constructor env name))) | ||||||||||
`(,(tc:constructor-entry-compressed-repr entry) | ||||||||||
,expr))) | ||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I get a bad error message for this example (coalton-toplevel
(repr :enum)
(define-type MyEnum A B C))
(coalton
(match A
(A "hi")
(B "bye"))) Because the patterns are read as (COMMON-LISP:LET ((#:MATCH4047
(COMMON-LISP:THE
(COMMON-LISP:MEMBER MYENUM/D MYENUM/C MYENUM/B MYENUM/A)
A)))
(COMMON-LISP:DECLARE (COMMON-LISP:IGNORABLE #:MATCH4047))
(COMMON-LISP:CASE A
(COMMON-LISP:OTHERWISE
(COMMON-LISP:LET ((A-3464 #:MATCH4047))
"hi"))
(COMMON-LISP:OTHERWISE
(COMMON-LISP:LET ((B-3465 #:MATCH4047))
"bye")))) I think there should be a check to ensure that a variable or There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. If we add this check (it makes sense) we should do this before codegen. |
||||||||||
(t | ||||||||||
`(otherwise (let ,bindings ,expr)))))) | ||||||||||
|
||||||||||
;; Only emit a fallback if there is not a catch-all clause. | ||||||||||
,@(unless (member-if (lambda (pat) | ||||||||||
(or (pattern-wildcard-p pat) | ||||||||||
(pattern-var-p pat))) | ||||||||||
(node-match-branches expr) | ||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. If a check is done to ensure that a variable or |
||||||||||
:key #'match-branch-pattern) | ||||||||||
`((otherwise | ||||||||||
(error "Fell through a non-exhaustive match.")))))))))) | ||||||||||
|
||||||||||
;; Otherwise do the thing | ||||||||||
(let ((subexpr (codegen-expression (node-match-expr expr) env)) | ||||||||||
(match-var (gensym "MATCH"))) | ||||||||||
|
@@ -206,7 +263,7 @@ | |||||||||
(node-match-branches expr) | ||||||||||
:key #'match-branch-pattern) | ||||||||||
`((t | ||||||||||
(error "Pattern match not exhaustive error"))))))))) | ||||||||||
(error "Fell through a non-exhaustive match."))))))))) | ||||||||||
|
||||||||||
(:method ((expr node-seq) env) | ||||||||||
(declare (type tc:environment env)) | ||||||||||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Currently, we only codegen if when
but not for
I think this simple change could allow both:
EDIT:
Actually, no, this won't work! In the main branch
returns "hi"... but so does this:
because
False
is being read as a variable!So actually, I think we should emit an "unused variable" error in this case. And, that's a separate issue not related to this PR.
EDIT no. 2:
I think this problem only creates confusion for Enums and Booleans, so your PR will actually fix this, and if this is merged, I don't think there is a reason to open a separate issue.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It's a long-standing issue that pattern variables can be confused as constructors by the programmer and we emit no good warning for this.