2017-08-29 18:38:51 +03:00
|
|
|
[![Build Status](https://travis-ci.org/ruby-x/rubyx.svg?branch=master)](https://travis-ci.org/ruby-x/rubyx)
|
2017-01-02 00:47:11 +02:00
|
|
|
[![Code Climate](https://codeclimate.com/github/ruby-x/rubyx/badges/gpa.svg)](https://codeclimate.com/github/ruby-x/rubyx)
|
|
|
|
[![Test Coverage](https://codeclimate.com/github/ruby-x/rubyx/badges/coverage.svg)](https://codeclimate.com/github/ruby-x/rubyx)
|
2014-05-30 14:49:34 +03:00
|
|
|
|
2017-01-02 00:29:20 +02:00
|
|
|
# RubyX
|
2014-04-14 05:51:44 -07:00
|
|
|
|
2017-01-02 00:47:11 +02:00
|
|
|
RubyX is about native code generation in and of ruby.
|
2018-03-20 15:40:22 +05:30
|
|
|
In other words, compiling ruby to binary, in ruby.
|
2017-01-02 00:47:11 +02:00
|
|
|
|
|
|
|
X can be read as 10 times faster, or a decade away, depending on mindset.
|
2014-04-14 16:46:17 +03:00
|
|
|
|
2018-03-20 15:40:22 +05:30
|
|
|
The current (fourth) rewrite adds aims at clarifying the roles of the different layers
|
|
|
|
of the system, see below. The overhaul is almost done.
|
2015-03-15 11:58:17 +00:00
|
|
|
|
2018-03-20 15:40:22 +05:30
|
|
|
Processing goes through layers: Ruby --> Vool --> Mom --> Risc --> Arm --> binary .
|
2017-01-02 00:47:11 +02:00
|
|
|
|
2015-03-15 11:58:17 +00:00
|
|
|
|
|
|
|
|
2018-03-20 15:40:22 +05:30
|
|
|
## Layers
|
2015-03-15 11:58:17 +00:00
|
|
|
|
2018-03-20 15:40:22 +05:30
|
|
|
### Ruby
|
2015-11-30 20:25:17 +02:00
|
|
|
|
2018-03-20 15:40:22 +05:30
|
|
|
Ruby is input layer, we use whitequarks parser to parse ruby and transform it to
|
|
|
|
Vool.
|
2015-11-30 20:25:17 +02:00
|
|
|
|
2018-03-20 15:40:22 +05:30
|
|
|
### Vool
|
2015-11-30 20:25:17 +02:00
|
|
|
|
2018-03-20 15:40:22 +05:30
|
|
|
Vool is a Virtual Object Oriented Language. Virtual in that is has no own syntax. But
|
|
|
|
it has semantics, and those are substantially simpler than ruby.
|
2015-11-30 20:25:17 +02:00
|
|
|
|
2018-03-20 15:40:22 +05:30
|
|
|
Vool is Ruby without the fluff. No unless, no reverse if/while, no splats. Just simple
|
|
|
|
oo. (Without this level the step down to the next layer was just too big)
|
2015-11-30 20:25:17 +02:00
|
|
|
|
2018-03-20 15:40:22 +05:30
|
|
|
Also Vool has a typed syntax tree, unlike the AST from the parser gem. This is easier when
|
|
|
|
writing conversion code: the code goes with the specific class (more oo than the visitor
|
|
|
|
pattern, imho)
|
|
|
|
|
|
|
|
### Mom
|
|
|
|
|
|
|
|
The Minimal Object Machine layer is the first machine layer. This means it has instructions
|
|
|
|
rather than statements. Instructions (in all machine layers) are a linked list.
|
|
|
|
|
|
|
|
Mom has no concept of memory yet, only objects. Data is transferred directly from object
|
|
|
|
to object with one of Mom's main instructions, the SlotLoad.
|
|
|
|
|
|
|
|
Mainly Mom is an easy to understand step on the way down. A mix of oo and machine. In
|
|
|
|
practise it means that the amount of instructions that need to be generated in vool
|
|
|
|
is much smaller (easier to understand) and the mapping down to risc is quite straightforward.
|
|
|
|
|
2018-03-20 15:52:16 +05:30
|
|
|
### Risc
|
2018-03-20 15:40:22 +05:30
|
|
|
|
|
|
|
The risc cpu architecture approach was a simplification of the cpu instruction set to a
|
|
|
|
minimum. Arm, our main target is a risc architecture, and the next level down.
|
|
|
|
|
2018-03-20 15:52:16 +05:30
|
|
|
The Risc layer here abstracts the Arm in a minimal and independent way. It does not model
|
|
|
|
any real RISC cpu instruction set, but rather implements what is needed for rubyx.
|
2018-03-20 15:40:22 +05:30
|
|
|
|
|
|
|
Instructions are derived from a base class, so the instruction set is extensible. This
|
2018-03-20 15:52:16 +05:30
|
|
|
way additional functionality may be added by external code.
|
2018-03-20 15:40:22 +05:30
|
|
|
|
|
|
|
Risc knows memory and has a small set of registers. It allows memory to register transfer
|
|
|
|
and back and inter register transfer. But has no memory to memory transfer like Mom.
|
|
|
|
|
|
|
|
### Arm
|
|
|
|
|
|
|
|
There is a minimal Arm assembler that transforms Risc instructions to Arm instructions.
|
|
|
|
This is mostly a one to one mapping, though it does introduce the quirks that ARM has
|
|
|
|
and that were left out of the Risc layer.
|
|
|
|
|
|
|
|
### Elf
|
|
|
|
|
|
|
|
Arm instructions assemble themselves into binary code. A minimal Elf implementation is
|
|
|
|
able to create executable binaries from the assembled code and Parfait objects.
|
|
|
|
|
|
|
|
### Parfait
|
|
|
|
|
|
|
|
Generating code (by descending above layers) is only half the story in an oo system.
|
|
|
|
The other half is classes, types, constant objects and a minimal run-time. This is
|
|
|
|
what is Parfait is.
|
|
|
|
|
|
|
|
## Types and classes, static vs dynamic
|
|
|
|
|
|
|
|
Classes in dynamic languages are open. They can change at any time, meaning you can add/remove
|
|
|
|
methods and use any instance variable. This is the reason dynamic languages are interpreted.
|
|
|
|
|
|
|
|
For Types to make any sense, they have to be static, immutable.
|
|
|
|
|
|
|
|
Some people have equated Classes with Types, this is a basic mistake in dynamic languages.
|
|
|
|
|
|
|
|
In rubyx a Type implements a Class (at a certain time of that classes lifetime). It
|
|
|
|
defines the methods and instance variables available. This is key to generating
|
|
|
|
efficient code that uses type information to access instance variables.
|
|
|
|
|
|
|
|
When a class changes, say a new method is added that uses a new instance variable, a
|
|
|
|
**new** Type is generated to describe the class at that point. **New** code is generated
|
|
|
|
for this new Type.
|
|
|
|
|
|
|
|
In essence the Class always **has a** current Type and **many** Types implement (different
|
|
|
|
versions of) a Class.
|
|
|
|
|
2018-03-20 15:52:16 +05:30
|
|
|
All Objects have a Type, as their first member (also integers!). The Type points to the
|
|
|
|
Class that the object has in oo terms.
|
2018-03-20 15:40:22 +05:30
|
|
|
|
|
|
|
Classes are defined by ruby code, but the methods of a Type (that are executed) are defined
|
|
|
|
by Mom and Risc only.
|
|
|
|
|
|
|
|
## Other
|
2015-10-07 11:32:48 +03:00
|
|
|
|
|
|
|
### Interpreter
|
|
|
|
|
|
|
|
After doing some debugging on the generated binaries i opted to write an interpreter for the
|
2018-03-20 15:52:16 +05:30
|
|
|
risc layer. That way tests run on the interpreter reveal most issues.
|
2015-10-07 11:32:48 +03:00
|
|
|
|
|
|
|
### Debugger
|
|
|
|
|
2017-08-29 18:38:51 +03:00
|
|
|
And after the interpreter was done, i wrote a [visual debugger](https://github.com/ruby-x/rubyx-debugger).
|
2018-03-20 15:52:16 +05:30
|
|
|
It is a simple opal application that nevertheless has proven a great help, both in figuring
|
|
|
|
out what is going on, and in finding bugs.
|
2015-05-17 20:07:52 +03:00
|
|
|
|
2015-10-07 11:32:48 +03:00
|
|
|
## Status
|
2015-05-17 20:07:52 +03:00
|
|
|
|
2018-03-20 15:40:22 +05:30
|
|
|
Just finishing the rewrite (above architecture) and about to get to binaries again.
|
2014-05-27 19:19:55 +03:00
|
|
|
|
2015-03-15 11:58:17 +00:00
|
|
|
### Stary sky
|
2014-05-30 14:49:34 +03:00
|
|
|
|
2015-03-15 11:58:17 +00:00
|
|
|
Iterate:
|
2014-05-30 14:49:34 +03:00
|
|
|
|
|
|
|
1. more cpus (ie intel)
|
|
|
|
2. more systems (ie mac)
|
|
|
|
3. more syscalls, there are after all some hundreds
|
|
|
|
5. A lot of modern cpu's functionality has to be mapped to ruby and implemented in assembler to be useful
|
|
|
|
6. Different sized machines, with different register types ?
|
|
|
|
7. on 64bit, there would be 8 bits for types and thus allow for rational, complex, and whatnot
|
|
|
|
8. Housekeeping (the superset of gc) is abundant
|
2014-09-19 19:39:08 +03:00
|
|
|
9. Any amount of time could be spent on a decent digital tree (see judy). Or possibly Dr.Cliffs hash.
|
|
|
|
10. Also better string/arrays would be good.
|
|
|
|
11. The minor point of threads and hopefully lock free primitives to deal with that.
|
2015-05-17 20:07:52 +03:00
|
|
|
12. Other languages, python at least, maybe others
|
|
|
|
13. translation of the vm instructions to another vm, say js
|
2014-05-30 14:49:34 +03:00
|
|
|
|
2018-03-20 15:52:16 +05:30
|
|
|
And generally optimise and work towards that perfect world (we never seem to be able to attain).
|
2014-05-30 14:49:34 +03:00
|
|
|
|
2014-04-16 12:45:36 +03:00
|
|
|
|
2014-04-14 16:46:17 +03:00
|
|
|
|
2018-03-20 15:52:16 +05:30
|
|
|
## Contributing to rubyx
|
2015-05-17 20:07:52 +03:00
|
|
|
|
2014-04-27 22:19:32 +03:00
|
|
|
Probably best to talk to me, if it's not a typo or so.
|
|
|
|
|
|
|
|
I do have a todo, for the adventurous.
|
|
|
|
|
|
|
|
Fork and create a branch before sending pulls.
|
2014-04-14 16:46:17 +03:00
|
|
|
|
|
|
|
== Copyright
|
|
|
|
|
2018-03-20 15:52:16 +05:30
|
|
|
Copyright (c) 2014-8 Torsten Ruger.
|
|
|
|
|
2015-11-30 20:25:17 +02:00
|
|
|
See LICENSE.txt for further details.
|