If one were to implement Ruby on top of a Javascript engine (either in the browser or on top of standalone V8 or Spidermonkey), what would be the key impedance mismatches between the Ruby and JS object models ?
If one were to implement Ruby on top of a Javascript engine (either in the browser or on top of standalone V8 or Spidermonkey), what would be the key impedance mismatches between the Ruby and JS object models ?
Share Improve this question asked Apr 1, 2011 at 15:53 Nick MainNick Main 1,4508 silver badges15 bronze badges 5- This question occurred to me while looking at the architecture of the SpiderMonkey tracing JIT - whether it could also serve as the guts of a Ruby implementation. It would also be nice to write Ruby for the browser (like GWT does for Java) - in that case I imagine that there would be many restrictions on the libraries and language features that could be used. – Nick Main Commented Apr 3, 2011 at 19:55
- I think that the biggest problem pointed out so far (by Marc-André Lafortune in a ment) is the missing method/const and autoload functionality. A future version of Ecmascript will contain "proxies" that would probably allow efficient implementation: wiki.ecmascript/doku.php?id=harmony:proxies – Nick Main Commented Apr 3, 2011 at 20:06
- Actually, I'm pretty sure proxies are in whatever the TraceMonkey version in Firefox 4 is, and I expect other implementors to follow suit, since they are immensely useful for security. (See, e.g. Mark S. Miller's SES (Secure ECMAScript) and Dr. SES (Distributed Resilient Secure ECMAScript) work). – Jörg W Mittag Commented Apr 4, 2011 at 1:26
-
The TraceMonkey tracing JIT consists mostly of
nanojit
, which is a language-independent JIT piler anyway, doesn't it? If all you want is a Ruby implementation with a tracing JIT, there is really no need to pile to ECMAScript, just hook upnanojit
to Rubinius. Or write a Ruby implementation in PyPy, whose tracing JIT is freakishly awesome. (Incidentally, PyPy has an ECMAScript backend, so you actually kill two birds with one stone. More like 4 birds, since a Ruby implementation written in PyPy would be able to run on natively, on the JVM, on .NET and on ECMAScript.) – Jörg W Mittag Commented Apr 5, 2011 at 2:00 - @Jörg thanks for the pointer to PyPy - that does look like an interesting project. – Nick Main Commented Apr 5, 2011 at 20:06
3 Answers
Reset to default 7The most in-your-face one is obviously the fact that ECMAScript is prototype-based and Ruby is class-plus-mixin-based. Also, in Ruby, encapsulation is done with objects, in ECMAScript with closures.
However, my guess is that Ruby's control flow constructs are going to be a much bigger hurdle than its object model. After all, James Coglan's JS.Class is basically an implementation of Ruby's object model in ECMAScript and it's not that big.
ECMAScript simply lacks the tools needed to build your own control-flow constructs on top of it. Typically, you need either GOTO
, continuations or proper tail calls. If you have one of those, you can easily implement everything else: exceptions, loops, switches, threads, Fiber
s, generators, coroutines, … you name it.
But ECMAScript doesn't have them (and for good reason, at least in the case of GOTO
). The only control-flow construct ECMAScript has that is powerful enough to be able to build other constructs on top of is exceptions. Unfortunately, those are pretty slow. (Nonetheless, they have been used as an implementation substrate, for example in the Microsoft Live Labs Volta piler, which used ECMAScript exceptions to implement .NET exceptions, iterators, generators and even threads.)
So, basically you are stuck with implementing at least your own call stack if not an entire interpreter (as is the case with HotRuby), performing global CPS transforms or something like that.
Basically, what you want from a Ruby engine running on top of ECMAScript, is
- a faithful implementation of the RubySpec (specifically the control-flow constructs such as threads, fibers,
throw
/catch
, exceptions etc.), - performance and
- tight integration with ECMAScript (i.e. the ability to pass objects and call methods back and forth between the two languages).
Unfortunately, when you have to resort to tricks like managing your own stack, doing CPS transforms, building on top of exceptions, … it turns out that you can only pick two of the three goals.
- Ruby has block-level scoping of local variables, JavaScript has function-level scoping
- Ruby's inheritance + mixins would likely be problematic to implement simply using JavaScript's prototypal inheritance
- Ruby's arity checking for method/lambda invocation and is more strict than JavaScript's permissive passing
- Ruby has true, enforced constants; JavaScript may not (depending on which version the interpreter is using)
- Class variables (ugh) have no equivalent in JaveScript, and so will require special handling
- Ruby's has green threads in the core, JavaScript does not
JavaScript is Turing-plete, so theoretically you can implement anything, including other programming languages in it. It doesn't matter how different the implementation (JavaScript) and the target language (Ruby) are. The impedance mismatch between languages like Ruby and C is huge, and there you have Ruby, Python, Perl, and JavaScript itself, all implemented in C.
Implementing Ruby in JavaScript should be orders of magnitude easier than doing so in a lower level language. To your advantage, you have that much of Ruby and Ruby's standard library are written in Ruby itself, so once you get a basic interpreter going, things should gradually be more and more downhill.
Implementing an efficient Ruby interpreter in JavaScript may be harder, but it's still possible. You may end up translating Ruby to JavaScript so the excellent optimizers available can kick in.
So, do not think about the differences between Ruby and JavaScript. Take a look at the standard implementation of Ruby, and think about how you would implement that in JavaScript.