1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
|
/* ScummVM - Graphic Adventure Engine
*
* ScummVM is the legal property of its developers, whose names
* are too numerous to list here. Please refer to the COPYRIGHT
* file distributed with this source distribution.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
*
* This program 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*
*/
%option noyywrap
%{
#define FORBIDDEN_SYMBOL_ALLOW_ALL
#include "common/str.h"
#include "director/lingo/lingo.h"
#include "director/lingo/lingo-gr.h"
int yyparse();
%}
identifier [_[:alpha:]][_[:alnum:]]*
constfloat [[:digit:]]+\.[[:digit:]]*
constinteger [[:digit:]]+
conststring \"[^\"\n]*\"
operator [-+*/%=^:,()><]
newline [ \t]*[\n\r]+
whitespace [\t ]
%%
--[^\r\n]*
^{whitespace}+
[\t]+ { return ' '; }
(?i:down) { return tDOWN; }
(?i:if) { return tIF; }
(?i:else[\t ]+if) { return tELSIF; }
(?i:else) { return tELSE; }
(?i:end) { return tEND; }
(?i:exit) { return tEXIT; }
(?i:frame) { return tFRAME; }
(?i:global) { return tGLOBAL; }
(?i:go) { return tGO; }
(?i:into) { return tINTO; }
(?i:loop) { return tLOOP; }
(?i:macro) { return tMACRO; }
(?i:mci) { return tMCI; }
(?i:mciwait) { return tMCIWAIT; }
(?i:movie) { return tMOVIE; }
(?i:next) { return tNEXT; }
(?i:of) { return tOF; }
(?i:previous) { return tPREVIOUS; }
(?i:put) { return tPUT; }
(?i:repeat) { return tREPEAT; }
(?i:set) { return tSET; }
(?i:then) { return tTHEN; }
(?i:to) { return tTO; }
(?i:with) { return tWITH; }
(?i:while) { return tWHILE; }
[!][=] { return tNEQ; }
[>][=] { return tGE; }
[<][=] { return tLE; }
{identifier} {
yylval.s = new Common::String(yytext);
if (Director::g_lingo->_builtins.contains(yytext))
return BLTIN;
return ID;
}
{constfloat} { yylval.f = atof(yytext); return FLOAT; }
{constinteger} { yylval.i = strtol(yytext, NULL, 10); return INT; }
{operator} { return *yytext; }
{newline} { return '\n'; }
{conststring} { yylval.s = new Common::String(&yytext[1]); yylval.s->deleteLastChar(); return STRING; }
.
%%
extern int yydebug;
namespace Director {
int Lingo::parse(const char *code) {
YY_BUFFER_STATE bp;
yydebug = 0;
yy_delete_buffer(YY_CURRENT_BUFFER);
bp = yy_scan_string(code);
yy_switch_to_buffer(bp);
yyparse();
yy_delete_buffer(bp);
return 0;
}
}
|