[Top][All Lists]
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]
[Chicken-janitors] Re: #439: Quasiquote simplification and improved synt
From: |
Chicken Trac |
Subject: |
[Chicken-janitors] Re: #439: Quasiquote simplification and improved syntax checks |
Date: |
Sun, 26 Dec 2010 23:48:55 -0000 |
#439: Quasiquote simplification and improved syntax checks
-----------------------------+----------------------------------------------
Reporter: sjamaan | Owner: felix
Type: change request | Status: new
Priority: minor | Milestone:
Component: expander | Version: 4.6.x
Resolution: | Keywords: expander, quasiquote, syntax
-----------------------------+----------------------------------------------
Comment(by felix):
I add here the possible options as given by Peter in his mail to chicken-
hackers:
{{{
Option 1:
As (quasiquote a b c) throws an error, so could
(quasiquote (quasiquote a b c)) or any even deeper-nested quasiquotes,
as well as unquotes. Having multiple arguments to quasiquote is
not allowed on level 0, so why should it be allowed on higher levels?
This is the pedantically correct option and might ease the debugging
of code that contains quasiquote calls because this code would be
unportable. This could break existing code.
This is probably the cleanest option from a language purity standpoint.
It would also make the code even a bit simpler than my patch does, I
think. We could put a call to ##sys#check-syntax after it's determined
that the car of a pair contains one of the three words quasiquote,
unquote or unquote-splicing. Then we have an if which checks the
nesting level and either calls WALK recursively or does the unquoting.
The disadvantage of this option compared to option 2 is that it
will be a little less practical when you want to use the literal
symbols quasiquote, unquote or unquote-splicing in deeper-nested
quotations.
With option 2, (quasiquote (x y (quasiquote a b c)))
will "just work". If this option would be implemented, this
would need to be written as
(quasiquote (x y (unquote-splicing (quote (quasiquote a b c)))))
or as
(quasiquote (x y (unquote (quote quasiquote)) a b c))
otoh, when written as `(x y ,'quasiquote a b c) that's not too
bad.
However, something like (quasiquote (x quasiquote y z)) would
also (perhaps unexpectedly) give an error as that's identical to
(quasiquote (x . (quasiquote y z)))
This difference cannot be detected because quasiquote, unlike normal
Scheme code, allows improper lists to occur anywhere. In normal Scheme
(x . (y z)) is identical to (x y z) and the y is never evaluated
as a macro.
Option 2 (this is what my current patch does):
Only throw errors in case of invalid list arguments at level 0
and increase the level with 1 whenever quasiquote is found at the
car of a pair. It will allow you to write quasiquotes and unquotes
at nesting levels above 1 without any strange trickery, and will
throw a nice error message when you try to use them at level 0 with
wrong arity.
The example above, (quasiquote (x quasiquote y z)), will just work
and not complain. However, (quasiquote (x quasiquote y z (unquote a)))
will not unquote the a, because this is identical to
(quasiquote (x . (quasiquote y z (unquote a)))), so (unquote a) occurs
at nesting level 1 and simply decreases the level to 0 instead of
evaluating the "a". That's the price we'd pay for having a loose and
non-strictly pedantic system.
Option 3:
Only fix R5RS behaviour and throw an error when an invalid number
of arguments are found at level 0. This will undo the part of my
patch that tries to make the behaviour more consistent. I am unsure
right now whether this is possible at all or if the inconsistencies
are what actually cause the r5rs breakage.
The main difference is that the code in quasiquote stays the same; it
will distinguish between quasiquote at the car of a pair which has a
pair as cdr and quasiquote at the car of a pair which has a non-pair
as cdr. Example: (quasiquote (quasiquote . #((unquote a))))
will result in trying to unquote a, whereas in options 1 and 2 it
will stay quoted because the second quasiquote increases the nesting
level. I can't think of any other cases than vectors where this
differs though (unless extra read syntax is added later).
Option 4:
Only add an error message when wrong number of arguments are found
at level 0. This would keep the R5RS-incompatibilities.
Option 5:
Do not change anything. It is not a bug, r5rs does not specify
what happens so silently dropping extra arguments to unquote is
acceptable.
Option 6:
Do not change the behaviour of unquote with multiple args, but do
fix the behaviour of nested quasiquotes to pass only the new
r5rs-compliance tests. This is option 3, but will leave the
behaviour that (quasiquote (quasiquote a b c)) => (quasiquote a)
Option 7:
Do it the R6RS way. We'd have to flesh out what exactly happens when
you do (quasiquote (quasiquote a b c)) or similar, and what happens when
you pass multiple arguments to unquote in a non-list context.
}}}
(I find it very hard to think of all the possible implications, but option
2 appears to me the easiest and least risky way)
--
Ticket URL: <http://bugs.call-cc.org/ticket/439#comment:4>
Chicken Scheme <http://www.call-with-current-continuation.org/>
Chicken Scheme is a compiler for the Scheme programming language.