Scripting Language
The DUELink Scripting Language that runs internally on any DUELink Hardware. The scripts are used to extend and tunnel in the commands from a Hosted Language such as Python, which is running on one of the Supported Systems such as Raspberry Pi. It can also be used to run the modules Standalone, independent from any host.
Scripts are not case sensitive, with a simple syntax that is inspired by BASIC and Python. The power of DUELink Scripts comes from its simplicity, rather than from its feature set. This is the perfect language to teach someone coding, to to extend DUELink modules with additional functionality.
Even though the scripting engine is not case sensitive, we use Print()
rather than print()
or prInT()
to keep things looking great!
Print
Print()
is a function that prints (outputs) the passed arguments. These arguments can be variables, strings, or equations. Print()
can handle multiple arguments.
PrintLn()
is exactly the same except it adds line break at the end.
x=100
PrintLn(x)
PrintLn("Hello World")
PrintLn(x+x)
PrintLn(x,"Hello World", x+x)
Print()
also supports printing arrays. When it sees a float array, such as Print(a1)
or Print({22.4,66.5})
the output will be a dump of the array values, {22.4,66.5}
in the later example. However, when it sees a byte array, it will print the ASCII values. Print("Hello DUELink") will show "Hello DUELink". This also applies to byte array variables and constants. Print([0x55])
will result in U
because 0x55
is ASCII for the letter U
.
This is an example on how to dump the values of a byte array, resulting in something similar to what print does with float arrays.
fn DumpBA(b0)
Print("[")
for i in range(Len(b0)
Print(b0[i],",")
next
Print("]")
fend
Comments
The #
character is used to identify a comment. Comments are ignored by the program, text added to help developers understand the code.
# This is a comment
x=10
Print(x) # This is also a comment
Whitespace
Space doesn't mean anything to the DUELink Scripting Language. However, new lines (or :
)are important to start new commands.
Print ( "DUELink" )
Is exactly the same as
Print("DUELink")
Variables
DUELink Script has a fixed set of 26 global float-type variables, one for each letter, assigned to _a
to _z
. To use a global variable, simply use _x=5.5
.
There are also local variables used in functions, still float-type. More on these under functions.
Arrays
There are 10 float-type arrays and 10 byte-type arrays. The float arrays are named a0 to a9 and the byte arrays are names b0 to b9. All arrays are size zero by default. Use Dim
to allocate memory for an array, like Dim b1[10]
.
Similar to other common languages, elements of an byte array are accessed using []
.
This is an example that uses both, global variables and arrays:
Dim b1[10]
For _i=0 to 9
b1[_i]=_i*2
Next
For _i=0 to 9
PrintLn(b1[_i])
Next
The output will look like:
0
2
4
6
8
10
12
14
16
18
Use Dim b1[0]
to free up the memory reserved for array b1[]
.
Same goes for float arrays.
Dim a1[10]
For _i=0 to 9
a1[_i]=_i* 0.3
Next
For _i=0 to 9
PrintLn(a1[_i])
Next
The output will look like:
0
0.3
0.6
0.9
1.2
1.5
1.8
2.1
2.4
2.7
Arrays can be initialized in two different ways.
Declare an array and initialize it with values at the same time.
Dim b1[6] = [1,2,
3,4,
5,6]
For _i in Range(Len(b1))
PrintLn(b1[_i])
Next
Len(b1)
returns the length (size) of the b1 array.
The second way is to create the array first, then populate it later.
Dim b1[6]
b1 = [7,8,
9,10,
11,12]
For _i in Range(Len(b1))
PrintLn(b1[_i])
Next
DUELink scripts use {}
for float arrays initializers, instead of []
which is used for byte arrays. The previous example will change to this.
Dim a1[6]
a1 = {7,8,
9,10,
11,12}
For _i in Range(Len(a1))
PrintLn(a1[_i])
Next
There are some specific things we must know about initializing arrays:
- Multi-line initializers must have a comma ending the line if the following line will have more data for the initializer (see the examples above).
- It is possible to start initialization values on a new line after
{
and then}
at the end of the last item. - Multi-line initializers can only be used in "record" mode. In immediate mode, the initializer must be on a single line.
- Initializers are always run, so if the initializer is inside a loop every time the dim or assignment initializer is encountered it will reinitialize the data in the array.
- You can have fewer values in the initializer than what the array holds, but you cannot have more. You will get an error indicating that a
]
was expected if there are too many elements in the initializer. - Since we do not want to do too many dynamic allocations, the size of the array must be specified when using dim even when initializing the array.
Dim
will automatically set the appropriate size when it sees an initializer, like `Dim a1[] = [1,2,3,4,5]' will create an arrays with 5 elements.
It is also possible to initialize an array with a string of text. The line dim b1[]="GHI"
will set b1[0] to ASCII G
and so on. The system automatically know if bytes or floats are needed and create the array properly. Note that this is not a string in its true meaning, but a simply way to initialize an array with ASCII characters.
A few escape codes are available with strings-like initializer, similar to the ones in the C language \r
, \n
, \t
, \"
. Print("He Said: \"I love DUELink\"")
will output He Said: "I love DUELink"
.
Functions that accept arrays can take initializers as well.
fn PrintArray(b1, c)
For _i in Range(c)
PrintLn(b1[_i])
next
fend
Dim b1[5] = [1,2,3,4,5]
PrintArray(b1, 3)
PrintArray([1,2,3,4,5], 3)
Functions
functions start with fn
and end with fend
. Arguments must be included in the function definition.
fn Add(a,b)
return a+b
fend
Then the function can be executed using Add()
. This can be done from withing the script or from an external source over one of the Interfaces.
Here it is used by the same script.
_x = Add(5,33)
PrintLn(_x)
Variables inside functions are local. You can use any of the 26 letters as a variable name. The argument will automatically set the local variable with the matching name to the passed value. In the previous example, a
and b
local variables were automatically assigned with first and second argument values.
Function arguments can also be arrays. These arrays are passed as by reference.
fn PrintArray(b1, c) # Note how the function expects the array is b1
For _i in Range(c)
PrintLn(b1[_i])
next
fend
Dim b3[] = [1,2,3,4,5]
PrintArray(b3, 3) # Note how we pass b3 array. The system will automatically reference b3 to b1, a1 used in the function.
Array initializer can also be used to call the function above.
PrintArray([1,2,3,4,5], 3) # Print 3 elements
And even use a string!
PrintArray("DUELink", 7)
Float array initializers use {}
as explained in the arrays section. Functions expecting float arrays do nto work with byte arrays, and vise vera.
Operands
DUELink Scripting supports the following operators.
Mathematical
Sign | Description |
---|---|
+ | Add |
- | Subtract |
* | Multiply |
/ | Divide |
% | Modulus, the remainder |
Comparators
Sign | Description |
---|---|
> | Greater Than |
< | Less Than |
>= | Greater Than or Equal To |
<= | Less Than or Equal To |
= | Equal |
!= | Not Equal |
Logical
Sign | Description |
---|---|
&& | And |
|| | Or |
Bitwise
Sign | Description |
---|---|
& | Bitwise And |
| | Bitwise Or |
^ | Bitwise Xor |
<< | Shift Right |
>> | Shift Left |
while-loop
This is the kind of loop that stays active as long as a condition is true. a while
loop block ends with wend
.
_x=10
while _x>5
Println(_x)
_x=_x-1
wend
For-Loop
The For-Loop has two different syntax styles. BASIC and Python style.
BASIC Style
The BASIC style For-Loop includes the last number in the range.
# Counting Up
For _i=1 to 5
Print(_i,",")
Next
Output:
1,2,3,4,
# Counting Up in increments of 10
For _i=1 to 1000 Step 10
PrintLn(_i)
Next
# Counting Down in increments of 10
For _i=1000 to 1 Step -10
PrintLn(_i)
Next
Python Style
The for-loop also allows a format similar to Python.
# Range with only stop value
For _i in range(5)
Print(_i,",")
next
Output:
0,1,2,3,4,
# Range with start and stop value
For _i in range(1,5)
Print(_i,",")
next
# Range with start, stop, and step value
For _i in range(1,5,2)
Print(_i,",")
next
# Range with start, stop, and negative step value
For _i in range(10,1,-2)
Print(_i,",")
next
If-Statement
If-Statements must end with the End
command.
If _x=1
PrintLn("one")
Else
PrintLn("not one")
End
If-Statements can also be nested within each other. Each If-Statement requires an End
command to terminate its own process.
If _x=1
PrintLn("one")
Else
If _x =2
PrintLn("two")
Else
PrintLn("not one or two")
End
End
Labels
Labels are needed to redirect the program. They are used by Goto
.
A Label is created by using the @
symbol in front of the desired label. Labels are limited to 8 characters.
Goto
Goto
is useful for repeating tasks indefinitely by sending execution to a specific Label name.
@Loop
# add code here that runs forever
Goto Loop
Exit
Exit
terminates the program.
Print("Hello")
Exit
Print("This will not get printed")
Multi-Command-Line
Multiple commands can be combined on a single line. This is especially useful when using Immediate mode
where a single line is required. To use multiple command, a :
symbol is used.
This is an example of a for loop in a single line
For _i=1 to 1000 Step 10:PrintLn(_i):Next
Aliases
Using aliases come allow for creating cleaner code. For example, the name temp
can be an alias to _t
variable holding temperature.
Alias()
is a special kind of function that adds new aliases to the system. It is not possible to remove an alias once it is added.
An alias can be used for variables, arrays, function names, and constants.
Dim b1[]=[65,66,67]
Alias(vals=b1, doit=test,age=23)
Println(age)
for i=0 to len(vals)-1
doit(vals[i], vals[i])
next
fn test(a, b)
Println(a," ",b)
fend
Aliases are handled in flash and has no RAM usage implications; however, they degrade performance due to the need for searching available aliases.