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.
playmoviealpharect was using the token for playmoviealpha, 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.tokenize was missing an entry in the token table, and so would not be recognised by the compiler.call "foo" syntax so that non-string constants will no longer be accepted.This is a modified copy of sslc, that has a few bugfixes from the original, that will recognise and compile the additional scripting 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 either get rid of the dos4gw.exe reference from p.bat or replace the original dos4gw.exe with the one in this archive.
If you use fallout script editor, you can extract compile.exe and dos4gw.exe to its \binary folder, or extract them somewhere else and change your preferences in FSE to point there. FSE doesn’t seem to be able to tell when errors occur when using this compiler though, so I’d recommend either using sfall’s script editor instead or compiling by hand if possible.
When compiling global or hook scripts for sfall 3.4 or below, you must include the line procedure start; before any #includes 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 safely (and is recommended) to be used with non-sfall scripts as well, as long as you don’t use any of the arrays syntax and any sfall scripting functions.
The original unmodified sslc source is over here: http://www.teamx.ru/site_arc/utils/index.html.
-q don't wait for input on error
-n no warnings
-b backward compatibility mode
-l no logo
-p preprocess source (don't generate .int)
-O optimize code (full by default, see optimization.md)
-O<N> set specific level of optimization (0 - off, 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
Необязательные аргументы в процедурах, определяемых пользователем. Вы можете использовать константы только для значений по умолчанию.
По сути, компилятор подставляет эти константы вместо опущенных аргументов.
// определение
procedure test(variable x, variable y := 0, variable z := -1) begin
...
end
...
// вызов
call test("value");
procedure test(variable x, variable y, variable z) begin
...
end
...
call test("value", 0, -1);
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 is null, the second condition will not be checked and your script won’t fail with “obj is null” error in debug.log
This also has an effect that a value of last computed argument is returned as a result of whole expressions, instead of always false (0) or true (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 the AND, OR operators in the script.
NOTE: Be aware that it may break some old scripts because operators behavior is changed slightly.
Conditional expressions, also known as ternary operator:
new:
// short style
x := if (condition) value1 : value2;
// python style
x := value1 if (condition) else value2
// combine style
x := if (condition) value1 : value2 if (condition) else value3 ...
if (condition) then
x := value1;
else
x := value2;
x = 5;
x := 5;
variable a, b, c;
variable begin a; b; c; end
variable tile := tile_num(dude_obj);
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.variable tile := (2 + 2);
0x to create a hexadecimal. The numbers 0 to 9 and A-F are allowed in the number. The number may not have a decimal point.a := 0x1000;
a := 4096;
++ and -- can be used as shorthand for += 1 and -= 1 respectively. They are mearly a syntactic shorthand to improve readability, and so their use is only allowed where += 1 would normally be allowed.a++;
a += 1;
for loops: Another piece of syntactic shorthand, to shorten while 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).
for (variable i := 0; i < N; i++) begin
display_msg("i = " + i);
end
// без инициализации счетчика
for (; i < N; i++) begin
display_msg("i = " + i);
end
i := 0;
while (i < N) do begin
display_msg("i = " + i);
i++;
end
NOTE: continue statement in a for 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.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 (see for and foreach sections for additional details).while (i < N) begin
// ...
if (condition) then break;
// ...
end
while (i < N and not(breakFlag)) begin
// ...
if (condition) then breakFlag := true;
// ...
end
for (i := 0; i < N; i++) begin
// ...
if (condition) then begin
// action
continue;
end
// else actions
end
for (i := 0; i < N; i++) begin
// ...
if (condition) then begin
// action
end else begin
// else actions
end
end
switch statements: A shorthand way of writing big if then else if... blocks
switch get_attack_type begin
case ATKTYPE_PUNCH: display_msg("punch");
case ATKTYPE_KICK: display_msg("kick");
default: display_msg("something else");
end
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
Оператор & (reference) предназначен для взятия ID-номера процедуры скрипта для передачи ее номера в аргумент функций.
Это в основном используется там где не предоставляется технической возможностью компилятора использовать имена процедур, в таких функциях как sfall_funcX.
sfall_func1(&MyProcedure);
Оператор @ (stringify) предназначен для того, чтобы сделать процедуры обратного вызовы лучше.
По сути, это заменяет имена процедур, которым предшествует @ их строковой константой.
callbackVar := @Node000;
call callbackVar;
callbackVar := "Node000";
call callbackVar;
NOTE: В Fallout вы можете вызывать процедуры, вызывая переменную, которая содержит имя самой процедуры в виде строкового значения.critical к процедуре).Empty statements in blocks are allowed: This is just a convenience to save scripters a bit of memory. Some of the macros in the fallout headers include their own semicolons while others do not. With the original compiler you had to remember which was which, and if you got it wrong the script would not compile. Now it’s always safe to include your own semicolon, even if the macro already had its own. For example, this would not compile with the original sslc, but will with the sfall edition:
#define my_macro diplay_msg("foo");
procedure start begin
my_macro;
end
NOTE: Does not work currently!
Используемый синтаксис ниже, требует наличия sfall для скомпилированных скриптов.
[] 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.md for more information)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
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
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:
// Temporary array
list := ["A", "B", "C", "D"];
// Permanent array
list := [*]["A", "B", "C", "D"];
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.md for full introduction to this type of arrays).map := {5: "five", 10: "ten", 15: "fifteen", 20: "twelve"};
. syntax to access elements of associative arrays and allows 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 of get_array calls twice.foreach loops: A shorthand method of looping over all elements in an array. Syntax is foreach (<symbol> in <expression>).
procedure bingle begin
foreach (variable 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 (<key symbol> : <value symbol> in <expression>)
foreach (variable 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 inside foreach.
The executation speed of scripts is not typically important in an unmodded game, given the difference in performance between a modern computer and what Fallout was designed for. When you start adding mods to the mix there’s the potential for problems again, since sfall’s global script system means that you can have a large amount of scripts being run every single frame.
The sfall build of sslc supports a -O command line option to perform an optimization pass over the generated code. This isn’t a magic make-my-code-go-faster bullet; most of what it does is very limited in scope. It’s primary purpose was to strip out the procedures and variables which get automatically pulled into every script that includes define.h, whether you use them or not, and to do something about the additional variables that get created by foreach loops.
There are several levels of optimization available:
-O1 - Basic, only removes unreferenced globals variables and procedures, code itself remains untouched.-O2 - Full, most code optimizations are on, but only those that were tested on complex scripts.-O3 - Experimental, provides most efficiency, but tend to break some complex code due to bugs.The following optimizations are performed:
a := 2 + 2; -> a := 4;
constant variable initialization: All variables are initialised to some value, (‘0’, if you don’t specify anything else,) so sslc attempts to make use of that fact to remove the first assignment to a variable if the first assignment is a constant expression.
variable a; -> variable a := 4;
a := 4; ->
constant propagation: checks for values assigned to variables which can be computed at compile time, and replaces relevent references to the symbol by the constant. The original store is not removed by this optimization. Global variables are considered for this optimization only if they are not marked import or export, and are not assigned to anywhere in the script.
a := 4; -> a := 4;
foo(a); -> foo(4);
dead code removal: Checks for and removes code which cannot be reached, either because it is hidden behind a return or because the argument to an if statement can be computed at compile time.
if (True) then begin -> display_msg("foo");
display_msg("foo"); ->
end else begin ->
display_msg("bar"); ->
end ->
unreferenced variable elimination: Checks for variables which are never referenced, and removes them. Also applies to global variables, as long as they are not marked for export.
variable i, j, k; -> variable i;
i := 1; -> i := 1;
return; -> return;
unreferenced procedure elimination: Checks for any procedures which are never called, and removes them.
procedure foo begin return "foo"; end -> procedure foo begin return "foo"; end
procedure bar begin return "bar"; end -> procedure start begin
procedure start begin -> display_msg(foo);
display_msg(foo); -> end
end ->
dead store removal: Removes variable assignments if the result of the variable is unused, and if the expression used to compute the value of the variable is provably free of side effects. (See ‘pure’ keyword)
a := "moo"; -> a := "foo";
a := "foo"; -> display_msg(a);
display_msg(a); ->
a := "bar"; ->
store combination: Where there are two stores in a row to the same variable, the two expressions are combined.
var1 := var2; -> var1 := var2 + var3;
var1 += var3; ->
variable combination: Where usage regions of variables do not overlap, combine the variables to provide additional candidates for unreferenced variable elimination. Very useful for scripts containing multiple foreach loops, which generate 2 or 3 hidden variables each.
a := "foo"; -> a := "foo";
display_msg(a); -> display_msg(a);
b := "bar"; -> a := "bar";
display_msg(b); -> display_msg(a);
namelist compression: Fallout stores the names of all file scope variables and procedures in a namelist which is saved into the .int. Any of these that are unreferenced can be removed, and the names of global variables can be modified to make them shorter.
Never concat constant strings with the ‘+’ operator, as it forces the operation to be done at runtime. The compiler can cope with constant strings being placed next to each other without the need for a +, which results in far more efficient code as the combination is done at lex time.
#define GLOB_PREFIX "ts__" -> #define GLOB_PREFIX "ts__"
procedure start begin -> procedure start begin
set_sfall_global(GLOB_PREFIX + "foo1", 0); -> set_sfall_global(GLOB_PREFIX "foo1", 0);
end -> end
Avoid function calls in while loops. function calls are expensive in comparison to variable lookups, so it’s more efficient to move the function call out of the loop and store the result in a variable
while i < len_array(array) do begin -> tmp := len_array(array);
... -> while i < tmp do begin
end -> ...
-> end
Mark functions with pure or inline where relevent.
pure is a hint to the optimizer that a procedure has no side effects. (i.e. there’s no way to tell that it’s been called aside from its return value.) Pure procedures cannot modify global variables, or call any other procedure that isn’t itself pure. Functions marked with pure can only be used in expressions (i.e. you cannot use the call <procedure> syntax to call them.) If there are non-pure terms in an expression, it prevents that expression being considered for dead store removal. Where no such optimizations can be performed, or if optimization is disabled, marking a procedure with pure will have no effect on the compiled code.
inline is an instruction to the compiler to replace calls to the marked procedure with a copy of the procedures code instead of having a seperate call. inlined procedures cannot use the ‘return’ command, cannot be predefined, and cannot be used as part of an expression. inlining if a procedure is only going to be called once is always a win, but if there are multiple calls to a procedure you will end up bloating the size of the generated code.
v1.5.1:
- добавлен опкод функции
get_perk_freq(sfall 5.1.3)
v1.5.0:
- оптимизация в построении компилируемого байт-кода скрипта (улучшение производительности выполнения скриптов)
- добавлен альтернативный тернарный оператор
x = if (condition) a : b;- добавлен дополнительный синтаксис
[*]для создания постоянного массива в выражений массивов (подробнее см. Array.md)- исправлена ошибка из-за которой не во всех участках тела процедуры можно было объявлять локальные переменные
- добавлена возможность объявлять переменные непосредственно в операторах
forиforeach.- добавлен дополнительный синтаксис для
forбез инициализации переменной счетчикаfor (; i < N; i++)- исправлена ошибка оператора
breakнеправильно работающий во вложенных циклах- добавлены опкоды функций
roundf,ceilf(sfall 5.1.1)- изменение в опции
-b(backward compatibility) компилирует скрипт в код совместимый с sfall v5.1.0 и ниже (прежний режим работы опции удален по причине неактуальности)- обновление версии декомпилятора int2ssl v8.6.0
v1.4.0:
- добавлен функция опкода
array_key_exists(sfall 5.0.5)- исправлена ошибка оптимизатора при удалении неиспользованных переменных в определении процедуры
- исправлена ошибка “symbol or string expected” при попытке вызова процедуры с использованием строкового литерала
- исправлена ошибка, когда оптимизатор не воспринимал вызов процедуры со строковой переменной как использование переменной
- исправлена ошибка из-за которой функция
get_arrayне работала с экспортированными переменными- исправлена ошибка из-за которой не работали необязательные аргументы для импортированных процедур
- исправлена ошибка при определении экспортируемой процедуры с аргументами
- исправлена сбоя компилятора, когда экспортируемая процедура не имела кода в теле процедуры
v1.3.1:
- добавлен оператор
&возвращающий номер процедуры скрипта- добавлен опкод
set_shader_type(sfall 5.0.3)
v1.3.0:
- добавлены опкоды
get_config_settingиset_config_setting(sfall 5.0.0)
v1.2.3:
- добавлены дополнительные опкоды
sfall_func7иsfall_func8(sfall 4.2.9)- исправлена ошибка компиляции скрипта при наличии маркера кодировки UTF-8 BOM
v1.2.2:
- добавлена возможность объявлять локальную переменную процедуры в любом месте тела процедуры
v1.2.0:
- добавлен новый оператор
divдля деления без знаковых целых чисел (sfall 4.2.3)- добавлены новые логические операторы
andAlso,orElseдля оценки логических выражений, без необходимости использования опции-s- добавлен альтернативный оператор присваивания как в языках C/Java
- исправлено сообщение ошибки компилятора “assignment operator expected” при неправильном определении макроса, похожего на переменную
v1.1.0:
- added new opcode
reg_anim_callback(sfall 4.2.2)- the basic optimization is now enabled by default when not specifying any optimization options
- unreferenced
criticalprocedures and procedures with the namesNode998andNode999are now removed by the optimizer
v1.0.0:
- added new opcode
register_hook_proc_spec(sfall 4.2.0)- added
-m<macro[=val]>option to define a macro named “macro” for conditional compilation- added
-I<path>option to specify an additional directory to search for include files- now it is possible to run preprocess or optimization passes in backward compatibility mode
SFALL 4.0: (нумерация версии компилятора не велась)
- enabled code for
ceilmath function- fixed missing argument for
how_muchfunction- added
desc_p_proc(from Fallout 1) to protected procedures that should not be removed by the optimizer- fixed compiler giving “division by zero” error when using zero as the second factor in multiplication
SFALL 3.8:
- added support for new universal opcodes
sfall_funcX
SFALL 3.6:
- added python-style ternary operator (conditional expression)
- added
-sshort-circuit evalution option forAND,ORexpressions- int2ssl will detect and decompile conditional expressions and short-circuit logical operators
- added
-Foption to include full file paths in#linedirectives after preprocessing- added
-Doption to write abstract syntax tree into.txtfile- fixed compiler crash when number of arguments in procedure declaration does not match definition
- fixed incorrect constant folding of
bwnotoperator- fixed more invalid results in constant folding
- implemented optional arguments for user-defined procedures
- implemented stringify procedure names using
@operator, which is helpful to pass procedures around to call them from variables (it will properly handle references)- logic for procedures passed as arguments to scripting functions was moved from code generation to parsing stage
- now it is possible to call user-defined procedures inside argument list of scripting functions, without compiler attempting to treat them as procedure references and probably fail (procedures will still be passed, but only to appropriate scripting functions at appropriate argument positions)
- int2ssl will now place empty parantheses after a call to user-defined procedure - this will distinct calls from passing procedures to some scripting functions (like giq_option)
- fixed
inlineprocedure “calls” not working when optimization is enabled- added column numbers to error/warning output
- added code to simplify adding sfall opcodes into compiler (need to add code in 3 places, instead of 7 places for each opcode)
- fixed bug when initializing variable with expression starting from a symbol
- added division by zero constant check
SFALL 3.5:
- completed namespace compression optimization with respect to imported/exported variables
- changed
forandforeachsyntax to allow parentheses which are easier to read IMHO- heavy code refactoring - split “parse.c” into several files, replaced all dirty workaround code in “lex()” (some syntax features) with parser-level equivalents
- added syntax to reference elements in multi-dimensional arrays (unlimited sequence of brackets
[]and dots.)- added fully featured
breakandcontinuestatements for loops- moved some optimizations (namely constant propagation and variable reuse) to “experimental” because they were breaking my scripts
- added ability to initialize variables with expressions
SFALL 3.4:
- added
foreach .. while ..syntax- added array expressions for lists and maps
- added
foreach (key : value in ...)syntax for convenience- fixed crash bug in
forloop parsing function- added ability to access array elements with string keys using OOP-like dot
.syntax
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:
for, foreach, break, continue, in and tokenize are now hardcoded names, existing scripts that use any of them as a variable or procedure name will no longer compile.addbuttonflag used to be recognised by the compiler, but would not emit any code into the int file.playmoviealpharect compiled as playmoviealpha.