SSLC
This is a modified copy of sslc, that has a few bugfixes from the original, that will recognise and compile the additional script functions provided by sfall, that can also understand some additional syntax, and that has an integrated preprocessor and optimizer.
Unlike the original script compiler, this has not been compiled as a DOS program. When using this in place of the original compile.exe
but still using p.bat
, you need to get rid of the dos4gw.exe
reference from p.bat
.
When compiling global or hook scripts for sfall 3.4 or below, you must include the line procedure start;
before any #include
files that define procedures to avoid a few weird problems. This is no longer required starting from 3.5.
This version of compiler was designed primarily for new sfall functions, but it can be safely used (and is recommended) with non-sfall scripts as well.
The original unmodified sslc source is over here: https://teamx.ru/site_arc/utils/index.html
Command line options
-q don't wait for input on error
-n no warnings
-b use backward compatibility mode
-l no logo
-p preprocess source
-P preprocess only (don't generate .int)
-F write full file paths in "#line" directives
-O optimize code (full optimization, see "Optimization" page)
-O<N> set specific level of optimization (0 - none, 1 - basic, 2 - full, 3 - experimental)
-d print debug messages
-D output an abstract syntax tree of the program
-o set output path (follows the input file name)
-s enable short-circuit evaluation for all AND, OR operators
-m<macro>[=<val>] define a macro named "macro" for conditional compilation
-I<path> specify an additional directory to search for include files
The original command line option -w
to turn on warnings no longer has an effect, since warnings are now on by default
Additional supported syntax
Syntax which requires sfall for compiled scripts to be interpreted is marked by asterisk (*).
-
Optional arguments in user-defined procedures. You can only use constants for default values. It basically puts those constants in place of omitted arguments.
- new:
procedure test(variable x, variable y := 0, variable z := -1) begin ... end ... call test("value");
- old:
procedure test(variable x, variable y, variable z) begin ... end ... call test("value", 0, -1);
- new:
-
New logical operators
AndAlso
,OrElse
for short-circuit evaluation of logical expressions. Using these operators allow the right part of logical expressions not to be evaluated (executed, computed) if the result is already known. This can improve the performance of running scripts.Example:
if (obj andAlso obj_pid(obj) == PID_STIMPAK) then ...
If
obj
isnull
, the second condition will not be checked and your script won’t fail with “obj is null” error in debug.logThis also has an effect that a value of last computed argument is returned as a result of whole expressions, instead of always
false
(0) ortrue
(1):obj := false; display_msg(obj orElse "something"); // will print "something"
You can also use the
-s
option to enable short-circuit evaluation for all theAND
,OR
operators in the script.NOTE: Be aware that it may break some old scripts because operators behavior is changed slightly.
- Conditional expressions (Python-inspired), also known as ternary operator:
- new:
X := value1 if (condition) else value2
- old:
if (condition) then X := value1; else X := value2;
- new:
- To assign values, you can use the alternative assignment operator from C/Java instead of Pascal syntax.
- new:
x = 5;
- old:
x := 5;
- new:
- Multiple variable declaration: Multiple variables can be declared on one line, separated by commas. This is an alternative to the ugly begin/end block, or the bulky single variable per line style.
- new:
variable a, b, c;
- old:
variable begin a; b; c; end
- new:
- Variable initialization with expressions: You can now initialize local variables with complex expressions instead of constants.
- new:
variable tile := tile_num(dude_obj);
- old:
variable tile; tile := tile_num(dude_obj);
NOTE: If your expression starts with a constant (eg.
2 + 2
), enclose it in parentheses, otherwise compiler will be confused and give you errors.
- new:
- Hexadecimal numerical constants: Simply prefix a number with
0x
to create a hexadecimal. The numbers 0 to 9 and letters A to F are allowed in the number. The number may not have a decimal point.- new:
a := 0x1000;
- old:
a := 4096;
- new:
- Increment/decrement operators:
++
and--
can be used as shorthand for+= 1
and-= 1
respectively. They are merely a syntactic shorthand to improve readability, and so their use is only allowed where+= 1
would normally be allowed.- new:
a++;
- old:
a += 1;
- new:
break
&continue
statements: They work just like in most high-level languages.break
jumps out of the loop.continue
jumps right to the beginning of the next iteration (seefor
andforeach
sections for additional details).- new:
while (i < N) begin // ... if (/* some condition */) then break; // ... end
- old:
while (i < N and not(breakFlag)) begin // ... if (/* condition */) then breakFlag := true; // ... end
- new:
for (i := 0; i < N; i++) begin // ... if (/* condition */) then begin // action continue; end // else actions end
- old:
for (i := 0; i < N; i++) begin // ... if (/* condition */) then begin // action end else begin // else actions end end
- new:
for
loops: Another piece of syntactic shorthand, to shortenwhile
loops in many cases. Parentheses around the loop statements are recommended but not required (when not using parentheses, a semicolon is required after the 3rd loop statement).- new:
for (i := 0; i < 5; i++) begin display_msg("i = " + i); end
- old:
i := 0; while (i < 5) do begin display_msg("i = " + i); i++; end
NOTE:
continue
statement in afor
loop will recognize increment statement (third statement in parentheses) and will execute it before jumping back to the beginning of loop. This way you will not get an endless loop.
- new:
switch
statements: A shorthand way of writing bigif then else if...
blocks- new:
switch get_attack_type begin case ATKTYPE_PUNCH: display_msg("punch"); case ATKTYPE_KICK: display_msg("kick"); default: display_msg("something else"); end
- old:
variable tmp; tmp := get_attack_type; if tmp == ATKTYPE_PUNCH then begin display_msg("punch"); end else if tmp == ATKTYPE_KICK then begin display_msg("kick"); end else begin display_msg("something else"); end
- new:
- Procedure stringify operator
@
: Designed to make callback-procedures a better option and allow for basic functional programming. Basically it replaces procedure names preceded by@
by a string constant.- old:
callbackVar := "Node000"; call callbackVar;
- new:
callbackVar := @Node000; call callbackVar;
Not many people know that since vanilla Fallout you can call procedures by “calling a variable” containing it’s name as a string value. There was a couple of problems using this:
- optimizer wasn’t aware that you are referencing a procedure, and could remove it, if you don’t call it explicitly (can be solved by adding making procedure
critical
) - you couldn’t see all references of a procedure from a Script Editor
- it was completely not obvious that you could do such a thing, it was a confusing syntax
- old:
- (*) Arrays: In vanilla Fallout, arrays had to be constructed by reserving a block of global/map variables. Since sfall 2.7, specific array targeted functions have been available, but they are fairly messy and long winded to use. The compiler provides additional syntactic shorthand for accessing and setting array variables, as well as for array creation. When declaring an array variable, put a constant integer in
[]
` to give the number of elements in the array. (before sfall 3.4 you had to specify size in bytes for array elements, now it’s not required, see “Arrays” page for more information)- new:
procedure bingle begin variable a[2]; a[0] := 5; a[a[0] - 4] := a[0] + 4; display_msg("a[0]=" + a[0] + ", a[1]=" + a[1]); end
- old:
procedure bingle begin variable a; a := temp_array(2, 4); set_array(a, 0, 5); set_array(a, get_array(a, 0) - 4, get_array(a, 0) + 4); display_msg("a[0]=" + get_array(a, 0) + ", a[1]=" + get_array(a, 1)); end
- new:
- (*) Array expressions: Sometimes you need to construct an array of elements and you will probably want to do it in just one expression. This is now possible:
- new:
list := ["A", "B", "C", "D"];
- old:
list := temp_array(4, 2); list[0] := "A"; list[1] := "B"; list[2] := "C"; list[3] := "D";
Syntax specific for associative arrays is also available. (see “Arrays” page for full introduction to this type of arrays).
- new:
- (*) Map array expressions:
map := {5: "five", 10: "ten", 15: "fifteen", 20: "twelve"};
- (*) The dot
.
syntax to access elements of associative arrays and allow to work with arrays like objects:trap.radius := 3; trap.tile := tile_num(dude_obj);
You can chain dot and bracket syntax to access elements of multi-dimensional arrays:
collectionList[5].objectList[5].name += " foo";
NOTE: When using incremental operators like
+=
,*=
,++
,--
compiler will use additional temp variable to get an array at penultimate level in order to avoid making the same chain ofget_array
calls twice. - (*)
foreach
loops: A shorthand method of looping over all elements in an array. Syntax isforeach (<symbol> in <expression>)
. You can declare variables in place by adding thevariable
keyword before the symbol name.- new:
procedure bingle begin variable critter; foreach (critter in list_as_array(LIST_CRITTERS)) begin display_msg("" + critter); end end
- old:
procedure bingle begin variable begin critter; array; len; count; end array := list_as_array(LIST_CRITTERS); len := len_array(array); count := 0; while count < len do begin critter := array[count]; display_msg("" + critter); end end
If you want an index array element (or key for “maps”) at each iteration, use syntax:
foreach (<symbol>: <symbol> in <expression>)
foreach (pid: price in itemPriceMap) begin if (itemPid == pid) then itemPrice := price; end
If you want to add additional condition for continuing the loop, use syntax:
foreach (<symbol> in <expression> while <expression>)
. In this case loop will iterate over elements of an array until last element or until “while” expression is true (whatever comes first).NOTE: Just like
for
loop,continue
statement will respect increments of a hidden counter variable, so you can safely use it insideforeach
. - new:
Fixes
playmoviealpharect
was using the token forplaymoviealpha
, breaking both functions in the process.addbuttonflag
had an entry in the token table, and could be parsed, but was missing an entry in the emit list. This resulted in the compiler accepting it as a valid function, but not outputting any code for it into the compiled script.- The function
tokenize
was missing an entry in the token table, and so would not be recognised by the compiler. - Fixed the check for the
call "foo"
syntax so that non-string constants will no longer be accepted.
Backward compatibility
There are several changes in this version of sslc which may result in problems for previously working scripts. A new command line option -b
is available, which will turn off all new fixes and features which have the possibility of causing a previously compiling script to change its behaviour. (And only those features; anything which would not compile under the old sslc is not affected.) This includes the following:
- Since
for
,foreach
,break
,continue
,in
andtokenize
are now hardcoded names, existing scripts that use any of them as a variable or procedure name will no longer compile. - Missing a semicolon after a variable declaration is now a hard error. (Originally sslc would check for the semicolon, but would not complain if it was missing.)
- The function
addbuttonflag
used to be recognised by the compiler, but would not emit any code into the int file. - The function
playmoviealpharect
compiled asplaymoviealpha
.
int2ssl note
int2ssl by Anchorite (TeamX) is included in sfall modderspack package. It was updated to support all additional opcodes of sfall, along with some syntax features. You can use it to decompile any sfall or non-sfall script.