Each C# file that is going to be translated should follow the listed rules:
- Class with Program methods must be marked with
[Program]
attribute - Each compiled .exe file must contain exactly one class with
[Program]
attribute. - Program class can inherit interfaces [future plans]
- Program class contains only public fields and public or private methods.
- Static fields and methods are not allowed.
- Public methods are translated to Program methods.
- Private methods are translated to inner functions, only difference between them and Program methods is that inner functions are not accessible from outside world and can only be called from Program method.
- Fields (that can be only public) are translated to storage items with
utf8("p_<field_name>")
keys. - Only one constructor of Program class is allowed. This constructor mustn't have any arguments.
User can freely define classes without [Program]
attribute.
Objects of these classes are translated to data struct
s and behave very similar to objects in C#.
Formally this translation follow the listed rules:
- Class doesn't have
[Program]
attribute,[Program]
attribute is used for only one class, which is translated to Program methods. - Interfaces are not translated, they serve only as compile-time entities.
- All (
private
,protected
,internal
andpublic
) fields are translated toutf8("<field_name>") -> <field_value>
pairs instruct
. - All methods are translated to
utf8("<method_name>_<args_types>")
->ref(#<inner_function_offset>)
. The<args_types>
prefix is needed to support overloading and<inner_function_offset>
means offset of function that should be called for the<method_name>
method of that class. Overridden methods will point to different<inner_function_offset>
. This technique is similar to Virtual method table. static
fields are translated to storage items withutf8("s_<class_name>_<field_name>")
keys.static
methods are translated to inner functions with<class_name>_<method_name>_<args_types>
names.- Constructors are translated to inner functions with
<class_name>_ctor_<args_types>
names.
Suppose we have the following classes definitions in C#:
interface Vehicle
{
void ComeIn(String someone);
}
class Bicycle : Vehicle
{
public String Owner = "no one";
void ComeIn(String someone) {
Owner = someone;
}
}
class Car : Vehicle
{
static private bool isBearBurnedDown = false;
static public bool IsBearBurnedDown()
{
return Car.isBearBurnedDown;
}
public int NumberOfTires;
public Car(int tires)
{
NumberOfTires = tires.
}
void ComeIn(String someone)
{
if (someone == "bear") {
Car.isBearBurnedDown = true;
}
}
}
interface Vehicle
won't be translated to anythingBicycle()
constructor will be translated toBicycle_ctor
function that createsstruct(utf8("Owner") -> utf8("no one"), utf8("ComeIn_string") -> ref(#<function1>)
Bicycle.ComeIn
will be translated to some function (let's call itfunction1
) that changesutf8("Owner")
field in the givenstruct
.Car(int)
constructor will be translated toCar_ctor_int32
function that createsstruct(utf8("NumberOfTires") -> int32(<given_int>), utf8("ComeIn_string") -> ref(#<function2>))
isBearBurnedDown
static field will be translated toutf8("s_Car_isBearBurnedDown") -> bool
storage itemIsBearBurnedDown
static method will be translated toCar_IsBearBurnedDown
function that readsutf8("s_Car_isBearBurnedDown")
storage key.Car.ComeIn
will be translated to some function (let's call itfunction2
) that changesutf8("s_Car_isBearBurnedDown") -> bool
storage item according to the givenString
from the stack.