-
Notifications
You must be signed in to change notification settings - Fork 26
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
Up for discussion: Move the "main runtime" of es out of C and into es script #79
base: master
Are you sure you want to change the base?
Conversation
This moves the bulk of the interesting stuff in main() to es:main. It also defines %dot and %batch-loop in es script. It does this by defining a new variable $runflags which operates like a modifiable version of $- in other shells.
- skip empty lines in $&batchloop - catch exceptions that happen during dumping
- Simplify $&access so that it doesn't need complex option parsing - Rewrite fn-access in initial.es to reproduce the same behavior as before - Add a %getopt function and migrate all the option parsing in initial.es/runtime.es to use it
It turns out there's actually a bit of convergent evolution here: the plan 9 rc s (both plan9port and 9front, but not Rakitzis rc) invoke a script called Also, all versions of rc have a |
This looks really cool, and definitely in keeping with es's vibe (i.e. that since you have a capable language as part of the shell, the C code recedes into the background as much as possible). It's tempting me to relax my stance on 'es is in maintenance mode, so let's not do anything too crazy', though I do feel like people 'with a strong understanding of es semantics' might be a fairly small group compared to those who understand C. |
I take that as quite the compliment :) FWIW, I'm obviously biased, but to my mind there's still a place for "crazy" changes to es, like the stuff Paul described in the mailing list as post-1.0. I don't think the "experiment" that es represents was ever really finished. But I don't have a strong enough understanding of github project management to say "let's shove it into this-or-that branch" or whatever.
Certainly the understatement of the last |
This is actually an unrelated mailing list suggestion (see http://wryun.github.io/es-shell/mail-archive/msg00977.html) which just happens to work very well in the context of a %main hook.
(I'm happy to let this change linger as a PR. I just wanted to share it with folks because I think it's really interesting. It's intended to be an invisible change for most users.)
So this started as a simple experiment asking "how much code, both in terms of LOC and 'interesting' behaviors, can you rip out of main.c?" It turns out that the answer is a LOT -- I was able to get main.c down to 50 lines, and those lines are nearly all just unconditional function calls. But the changes to enable that are a lot more interesting.
Essentially, what happened here is that the logical flow of es, from setup to REPL, has been inverted from a "C code that calls into es script for user customizability" model to "es script that calls into C code as its execution platform." To illustrate what I mean, look at the basic logical startup flow when executed as an interactive login shell.
In "normal" es, it looks something like this:
Most of of the interesting stuff happens in
main.c
andinput.c
, with just brief detours into hook functions.In this experimental es, here's how it goes:
Nearly everything in this flow has moved into the new
runtime.es
file, which is sourced byinitial.es
at dump time. All the argument parsing, startup logic, the invocation of the REPLs, the definition of%dispatch
, are here. A person with a strong understanding of es semantics doesn't need to touch a single line of C to follow the path. They could even do something like make a new%login-loop
REPL and invoke it when the shell is a login shell (by editing%run-file
and%run-string
), or change how%dispatch
works (by editingset-runflags
), or create a new.bashrc
-style "on interactive startup" script (by editinges:main
), all withinruntime.es
. This is, to me, very exciting -- talk about an extensible shell!This is also at least somewhat related to the direction Paul Haahr wanted to go in, as excerpted from this mailing list post:
There are a couple other novelties here that I also find very interesting. The first is the
$runflags
variable. This variable is how the-eivxnlLGI
flags are implemented (and could be how more flags are implemented as well). It corresponds somewhat with the$-
variable in some other shells, but because of some of the strengths of es, instead of being formatted as a sneeze-like "himBHs", it reads likeinteractive login
. So, for example, if a bit of code wants to know if it is running in an interactive context, it tests~ $runflags interactive
-- in fact, the%is-interactive
function has been rewritten to just this test.Something that differentiates
$runflags
, though, is that because it works viaset-runflags
and$&setrunflags
, it can be changed in the middle of shell execution. This means a user can runrunflags = $runflags echoinput
, and the shell will start behaving as if it had been called with-v
. A script could also, to protect critical sections, do this:Combined with the exception-based exitonfalse behavior in #73, this seems like it could make for a powerful, flexible error-handling pattern.
One other thing that has been added for this change is dump-time-only primitives. The primitives defined in the new file
prim-dump.c
are only available in theesdump
binary, not thees
binary. I've just used this so far as a way to define a bootstrap$&batchloop
that doesn't need to be in thees
binary, and to inject some conditional compilation intoinitial.es
. I suspect that it could be used for more, though.All in all, this change is not really a shell-size win, measured either in LOC (though there are some fewer lines of C, and even more if you discount the lines that are exclusively used in
esdump
now), or in binary size (in general, to implement the same behavior in es and C, the former will require a bigger binary). But neither change is very large. There is presumably also a negative performance impact, but I haven't profiled it and, just as an interactive user, it isn't as large as one might expect.Follow-up edit: Perhaps I've gone too far now, but now all the option parsing is out of the C code and moved to a new pure-es
%getopt
function. This provides consistent behavior between%main
,%dot
,access
, andvars
. This was enabled by simplifying$&access
, the last user ofopt.c
, and reimplementing the complicated, option-parsing stuff in initial.es.