Initial script lexer, parser and compiler support.

This commit is contained in:
Loïc Hoguin 2010-11-17 01:30:07 +01:00
parent 71772a58fb
commit 7fadf362b3
4 changed files with 156 additions and 1 deletions

View File

@ -18,7 +18,8 @@
%% along with EGS. If not, see <http://www.gnu.org/licenses/>.
-module(egs_files).
-export([load_counter_pack/2, load_quest_xnr/1, load_table_rel/1, load_text_bin/1, load_unit_title_table_rel/2, nbl_pack/1, nbl_padded_size/1]).
-export([load_counter_pack/2, load_quest_xnr/1, load_script_bin/1, load_table_rel/1,
load_text_bin/1, load_unit_title_table_rel/2, nbl_pack/1, nbl_padded_size/1]).
%% @doc Build a counter's pack file, options and return them along with the background value.
load_counter_pack(ConfFilename, CounterNbl) ->
@ -208,6 +209,13 @@ load_quest_xnr_zones([Zone|Tail], BasePos, SetsAcc, SetsPtrsAcc, ZonesAcc) ->
ZoneBin = << ZoneID:16/little, AreaID:16/little, AreaID:32/little, 0:16, EnemyLevel:8, 16#ff:8, 16#04010000:32, SetPos:32/little, 0:352 >>,
load_quest_xnr_zones(Tail, BasePos + byte_size(SetsBin2), [SetsBin2|SetsAcc], [SetPos|SetsPtrsAcc], [ZoneBin|ZonesAcc]).
%% @doc Load a script file and compile it into the bytecode used by the game.
load_script_bin(ScriptFilename) ->
{ok, Script} = file:read_file(ScriptFilename),
{ok, Tokens, _NbLines} = egs_script_lexer:string(binary_to_list(Script)),
{ok, ParseTree} = egs_script_parser:parse(Tokens),
egs_script_compiler:compile(ParseTree).
%% @doc Load a counter configuration file and return a table.rel binary along with its pointers array.
load_table_rel(ConfFilename) ->
{ok, Settings} = file:consult(ConfFilename),

View File

@ -0,0 +1,59 @@
%% @author Loïc Hoguin <essen@dev-extend.eu>
%% @copyright 2010 Loïc Hoguin.
%% @doc EGS script compiler.
%%
%% This file is part of EGS.
%%
%% EGS is free software: you can redistribute it and/or modify
%% it under the terms of the GNU Affero General Public License as
%% published by the Free Software Foundation, either version 3 of the
%% License, or (at your option) any later version.
%%
%% EGS is distributed in the hope that it will be useful,
%% but WITHOUT ANY WARRANTY; without even the implied warranty of
%% MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
%% GNU Affero General Public License for more details.
%%
%% You should have received a copy of the GNU Affero General Public License
%% along with EGS. If not, see <http://www.gnu.org/licenses/>.
-module(egs_script_compiler).
-export([compile/1]).
%% @doc Compile a script parsed using egs_script_lexer and egs_script_parser.
compile(ParseTree) ->
RootBin = root(ParseTree),
FooterPos = byte_size(RootBin),
FooterSize = 0, %% @todo
<< $T, $S, $B, $2, FooterPos:32/little, FooterSize:32/little, RootBin/binary >>.
root(Routines) ->
root(Routines, 0, []).
root(nil, _Pos, Acc) ->
iolist_to_binary(lists:reverse(Acc));
root({Routine, Next}, Pos, Acc) ->
Bin = routine(Routine),
Pos2 = case Next of nil -> 0; _ -> Pos + byte_size(Bin) + 4 end,
root(Next, Pos2, [<< Pos2:32/little, Bin/binary >>|Acc]).
routine({Type, Name, Instructions}) ->
TypeBin = case Type of event -> << $E, $V, $E, $N, $T, $. >>; function -> << >> end,
NameBin = list_to_binary(Name),
Padding = 8 * (32 - byte_size(TypeBin) - byte_size(NameBin)),
InstrsBin = instructions(Instructions),
InstrsSize = byte_size(InstrsBin) + 4,
VarsBin = << 76:32/little, 0:32/little >>, %% @todo 0 vars for now.
<< TypeBin/binary, NameBin/binary, 0:Padding,
InstrsSize:32/little, VarsBin/binary, InstrsBin/binary, 0:32 >>.
instructions(Instrs) ->
instructions(Instrs, []).
instructions(nil, Acc) ->
iolist_to_binary(lists:reverse(Acc));
instructions({Instr, Next}, Acc) ->
instructions(Next, [instruction(Instr)|Acc]).
instruction({push, N}) when is_integer(N) ->
<< 2:32/little, N:32/little >>;
instruction({syscall, N}) when is_integer(N) ->
<< 97:32/little, N:32/little >>.

50
src/egs_script_lexer.xrl Normal file
View File

@ -0,0 +1,50 @@
%% @author Loïc Hoguin <essen@dev-extend.eu>
%% @copyright 2010 Loïc Hoguin.
%% @doc EGS script lexer.
%%
%% This file is part of EGS.
%%
%% EGS is free software: you can redistribute it and/or modify
%% it under the terms of the GNU Affero General Public License as
%% published by the Free Software Foundation, either version 3 of the
%% License, or (at your option) any later version.
%%
%% EGS is distributed in the hope that it will be useful,
%% but WITHOUT ANY WARRANTY; without even the implied warranty of
%% MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
%% GNU Affero General Public License for more details.
%%
%% You should have received a copy of the GNU Affero General Public License
%% along with EGS. If not, see <http://www.gnu.org/licenses/>.
Definitions.
D = [0-9]
L = [a-z]
N = ({L}|{D}|_)
WS = ([\000-\s]|%.*)
Rules.
{D}+ : {token, {integer, TokenLine, list_to_integer(TokenChars)}}.
{N}+ : {token, case reserved_word(TokenChars) of
false -> case syscall(TokenChars) of
false -> {name, TokenLine, TokenChars};
Syscall -> {syscall, TokenLine, Syscall}
end;
KeyWord -> {KeyWord, TokenLine}
end}.
-> : {token,{'->', TokenLine}}.
[}{,] : {token, {list_to_atom(TokenChars), TokenLine}}.
\.{WS} : {end_token, {dot, TokenLine}}.
{WS}+ : skip_token.
Erlang code.
reserved_word("event") -> event;
reserved_word("function") -> function;
reserved_word("push") -> push;
reserved_word(_) -> false.
syscall("play_music") -> 253;
syscall(_) -> false.

38
src/egs_script_parser.yrl Normal file
View File

@ -0,0 +1,38 @@
%% @author Loïc Hoguin <essen@dev-extend.eu>
%% @copyright 2010 Loïc Hoguin.
%% @doc EGS script parser.
%%
%% This file is part of EGS.
%%
%% EGS is free software: you can redistribute it and/or modify
%% it under the terms of the GNU Affero General Public License as
%% published by the Free Software Foundation, either version 3 of the
%% License, or (at your option) any later version.
%%
%% EGS is distributed in the hope that it will be useful,
%% but WITHOUT ANY WARRANTY; without even the implied warranty of
%% MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
%% GNU Affero General Public License for more details.
%%
%% You should have received a copy of the GNU Affero General Public License
%% along with EGS. If not, see <http://www.gnu.org/licenses/>.
Nonterminals declarations declaration instructions instruction.
Terminals integer name event function push syscall '->' ',' dot.
Rootsymbol declarations.
declarations -> declaration declarations : {'$1', '$2'}.
declarations -> '$empty' : nil.
declaration -> event name '->' instructions dot : {event, unwrap('$2'), '$4'}.
declaration -> function name '->' instructions dot : {function, unwrap('$2'), '$4'}.
instructions -> instruction ',' instructions : {'$1', '$3'}.
instructions -> instruction : {'$1', nil}.
instruction -> push integer : {push, unwrap('$2')}.
instruction -> syscall : {syscall, unwrap('$1')}.
Erlang code.
unwrap({_,_,V}) -> V.