RubimC
It is a Ruby compiler and framework for microcontrollers. Full name RubimCode in Russian transcription heard as "cut down the code". Current version is working but realizes far not all the features of Ruby. All realized features you can find in folder "examples"
Description:
RubimC designed to simplify the process of programming microcontrollers, but can also be used as an clear С-code generator. The framework is a syntax-flavored Ruby combines the unique features of the Ruby, adding and expanding the functions required for a specific area. At the input generator takes the program to Ruby, and the output provides a pure C code, based on the user program and libraries that are connected to a select model of the microcontroller. All that is required to generate is installed ruby-interpreter. If it`s nessesary, for compile C-code you need installing compiler gcc or avr-gcc (for AVR microcontrollers)
Benefits of writing programs in RubymC
- increase development speed
- code readability and elegance inherent in the Ruby language
- an object-oriented approach
- the use of an interpreted language does not reduce the performance of the final program because there is no virtual mashine
- ability of hardware control IC and delivery of messages
- ability to get a list of the hardware for a particular version of the device, as well as a list of all methods and help them directly from the generator console, on the basis of libraries.
Why?
First of all for fan...I want to see at Ruby from other point, not only from famous framework Ruby On Rails. Of course, we have great project mruby, that compile Ruby code, realized all common functions of Ruby and standart libraries, and supported by Matz. But...mruby generate a big-size code, and, as we know, microcontroller have very small memory. For example, for initialize only one array mruby generate binary file with size 1MB! At the other side RubimC generate code with minimal size, in most cases are not different from the similar size of, written on C. In addition, RubimC generator is clearness. You always can to see on generated C-code and to evaluate its performance and size.
How it`s work
Code generated in three stage:
- Preprocessing user programm, that replaced some Ruby keywords, operators and identificators
- Shell Ruby-code and generate C-code (use metaprograming of Ruby)
- Compile C-code (with gcc or avr-gcc).
Install
All you need to use RubimC gem is Ruby interpretator and gcc/avr-gcc compiler.
How to install Ruby. For Ubuntu I recomended to use first chapter of this manual.
Then you can install RubimC gem:
gem install rubimc
Compiler gcc provided by Linux. For others platforms use this manual.
To install avr-gcc use this manual.
ToDo list (main of them):
- Validate code before preprocessing
- Support C-libraries
- Code generator:
- define user`s variables as local (now it defined in top-level instance)
- support all C types of variables (int ,float, unsigned int, u_int8 e.t.)
- check match types (in assign and operations) and try to cast
- support array, hash, string, range and constants (as full as possible)
- support user`s methods and classes
- support threads
- Write libraries for microcontrollers (AVR, PIC, STM, e.t.)
- Fix a lot of possible bugs & features
What is done now
- Initialize variables (supported types: bool,int,float,double)
- Support most of ruby operators: arithmetic, unary, comparison, binary (need to realize logical operators: and/or/not)
- Support conditions if/unless (with return values) and it modify version
- Suppotr loops while/until and it modify version (except redo/retry instruction)
- Realize example library for AVR AtTiny13 MCU with DigitalIO and ADC support
Example for AVR microcontroller:
Ruby programm ("FirstController.rb"):
require 'rubimc'
class FirstController < AVR_attiny13
def initialize
ANALOG_TO_DIGITAL.init(ref: "vcc", channel: ADC0)
ANALOG_TO_DIGITAL.interrupt(enabled: true) do |volts|
output :led, port: :B, pin: 3
led.off if volts < 30
led.on if volts >= 220
end
end
def main_loop # infinit loop, it stop only when MCU is reset
end
end
To compile this code run in console:
rubimc compile FirstController.rb
or just
rubimc compile --all
It generate C-code placed in "release/FirstController.c" and hex-file for upload in MCU placed in "release/FirstController.hex
//=============================
#include <stdbool.h>
#define F_CPU 1000000UL
#include <avr/io.h>
#include <avr/iotn13.h>
#include <avr/interrupt.h>
int main()
{
// Init ADC
ADMUX = (0<<REFS0) | (0<<ADLAR) | MUX0;
ADCSRA = (1<<ADEN) | (0<<ADATE) | (1<<ADPS0) | (0<<ADIE);
ADMUX |= 1<<ADIE;
return 1;
}
// ADC Interrupt
ISR(ADC_vect)
{
int __rubim__volt = ADCL + ((ADCH&0b11) << 8);
DDRB |= 1<<(3);
if ((__rubim__volt<=(0))) {
PORTB &= 255 ^ (1<<(3));
}
if (!((__rubim__volt<(15)))) {
PORTB |= 1<<(3);
}
}
note: this is a valid C-code, but in real AVR-controllers it may not work, because avr-libraries are still in development
Some rake helpers
For create new project RubimC gem support command "generate" (of just "g"). For example:
rubimc generate mcu BrainControll type:attiny13 # create template "BrainControll.rb" for AVR microcontroller 'attiny13'
rubimc g mcu FirstProg # create template "FirstProg.rb" for unknown microcontroller
rubimc g clearC Example # create template "Example.rb" for generate clear C code
Some interesting idea
There is interesting idea for connect few microconrollers (IC) via some firmware interfaces, for example I2C or USB (at this moment is not realized). This example will generate two binary files for each microcontroller.
class BrainController < AVR_atmega16
def initialize()
input :@button, port: :A, pin: 6
end
def main_loop() # infinit loop, it stop only when IC is reset
if want_to_say_hello?
LeftHandController.move_hand = "up" # transfer data to other controller
end
end
private # define user`s methods and classes
def want_to_say_hello?
.press?
end
end
class LeftHandController < AVR_attiny13
attr_accessor :move_hand, via: "I2C" # I2C - fireware data-transfer bus
def move_hand=() # execute when command is received
@led.toggle
if =="up"
# ...run motor...
end
end
def initialize()
output :@led, port: :B, pin: 3
end
def main_loop() # infinit loop, it stop only when IC is reset
end
end
Help
If you interested the project and find some bugs, you may to write the tests and we try to fix it. Examples of tests is placed in folder test. To run tests use command "rspec test/test_all.rb". Of course if you try to modify core and libraries it will be wonder.
Thank you!