mirror of
https://github.com/ZDoom/qzdoom.git
synced 2025-01-10 03:30:44 +00:00
159 lines
3.6 KiB
ReasonML
159 lines
3.6 KiB
ReasonML
|
/* re2c lesson 001_upn_calculator, calc_008, (c) M. Boerger 2006 - 2007 */
|
||
|
/*!ignore:re2c
|
||
|
|
||
|
- using -b with signed character input
|
||
|
. Since the code is being generated with -b switch re2c requires the internal
|
||
|
character variable yych to use an unsigned character type. For that reason
|
||
|
the previous lessons had a conversion at the beginning of their scan()
|
||
|
function. Other re2c generated code often have the scanners work completely
|
||
|
on unsigned input. Thus requesting a conversion.
|
||
|
|
||
|
To avoid the conversion on input, re2c allows to do the conversion when
|
||
|
reading the internal yych variable. To enable that conversion you need to
|
||
|
use the implace configuration 're2c:yych:conversion' and set it to 1. This
|
||
|
will change the generated code to insert conversions to YYCTYPE whenever
|
||
|
yych is being read.
|
||
|
|
||
|
- More inplace configurations for better/nicer code
|
||
|
. re2c allows to overwrite the generation of any define, label or variable
|
||
|
used in the generated code. For example we overwrite the 'yych' variable
|
||
|
name to 'curr' using inplace configuration 're2c:variable:yych = curr;'.
|
||
|
|
||
|
. We further more use inplace configurations instead of defines. This allows
|
||
|
to use correct conversions to 'unsigned char' instead of having to convert
|
||
|
to 'YYCTYPE' when placing 're2c:define:YYCTYPE = "unsigned char";' infront
|
||
|
of 're2c:yych:conversion'. Note that we have to use apostrophies for the
|
||
|
first setting as it contains a space.
|
||
|
|
||
|
. Last but not least we use 're2c:labelprefix = scan' to change the prefix
|
||
|
of generated labels.
|
||
|
*/
|
||
|
|
||
|
#include <stdlib.h>
|
||
|
#include <stdio.h>
|
||
|
#include <string.h>
|
||
|
|
||
|
#define DEBUG(stmt) stmt
|
||
|
|
||
|
int stack[4];
|
||
|
int depth = 0;
|
||
|
|
||
|
int push_num(const char *t, const char *l, int radix)
|
||
|
{
|
||
|
int num = 0;
|
||
|
|
||
|
if (depth >= sizeof(stack))
|
||
|
{
|
||
|
return 3;
|
||
|
}
|
||
|
|
||
|
--t;
|
||
|
while(++t < l)
|
||
|
{
|
||
|
num = num * radix + (*t - '0');
|
||
|
}
|
||
|
DEBUG(printf("Num: %d\n", num));
|
||
|
|
||
|
stack[depth++] = num;
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int stack_add()
|
||
|
{
|
||
|
if (depth < 2) return 4;
|
||
|
|
||
|
--depth;
|
||
|
stack[depth-1] = stack[depth-1] + stack[depth];
|
||
|
DEBUG(printf("+\n"));
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int stack_sub()
|
||
|
{
|
||
|
if (depth < 2) return 4;
|
||
|
|
||
|
--depth;
|
||
|
stack[depth-1] = stack[depth-1] - stack[depth];
|
||
|
DEBUG(printf("+\n"));
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int scan(char *p)
|
||
|
{
|
||
|
char *t;
|
||
|
int res = 0;
|
||
|
|
||
|
while(!res)
|
||
|
{
|
||
|
t = p;
|
||
|
/*!re2c
|
||
|
re2c:define:YYCTYPE = "unsigned char";
|
||
|
re2c:define:YYCURSOR = p;
|
||
|
re2c:variable:yych = curr;
|
||
|
re2c:indent:top = 2;
|
||
|
re2c:yyfill:enable = 0;
|
||
|
re2c:yych:conversion = 1;
|
||
|
re2c:labelprefix = scan;
|
||
|
|
||
|
DIGIT = [0-9] ;
|
||
|
OCT = "0" DIGIT+ ;
|
||
|
INT = "0" | ( [1-9] DIGIT* ) ;
|
||
|
WS = [ \t]+ ;
|
||
|
|
||
|
WS { continue; }
|
||
|
OCT { res = push_num(t, p, 8); continue; }
|
||
|
INT { res = push_num(t, p, 10); continue; }
|
||
|
"+" { res = stack_add(); continue; }
|
||
|
"-" { res = stack_sub(); continue; }
|
||
|
"\000" { res = depth == 1 ? 0 : 2; break; }
|
||
|
[^] { res = 1; continue; }
|
||
|
*/
|
||
|
}
|
||
|
return res;
|
||
|
}
|
||
|
|
||
|
int main(int argc, char **argv)
|
||
|
{
|
||
|
if (argc > 1)
|
||
|
{
|
||
|
char *inp;
|
||
|
int res = 0, argp = 0, len;
|
||
|
|
||
|
while(!res && ++argp < argc)
|
||
|
{
|
||
|
inp = strdup(argv[argp]);
|
||
|
len = strlen(inp);
|
||
|
if (inp[0] == '\"' && inp[len-1] == '\"')
|
||
|
{
|
||
|
inp[len - 1] = '\0';
|
||
|
++inp;
|
||
|
}
|
||
|
res = scan(inp);
|
||
|
free(inp);
|
||
|
}
|
||
|
switch(res)
|
||
|
{
|
||
|
case 0:
|
||
|
printf("Result: %d\n", stack[0]);
|
||
|
return 0;
|
||
|
case 1:
|
||
|
fprintf(stderr, "Illegal character in input.\n");
|
||
|
return 1;
|
||
|
case 2:
|
||
|
fprintf(stderr, "Premature end of input.\n");
|
||
|
return 2;
|
||
|
case 3:
|
||
|
fprintf(stderr, "Stack overflow.\n");
|
||
|
return 3;
|
||
|
case 4:
|
||
|
fprintf(stderr, "Stack underflow.\n");
|
||
|
return 4;
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
fprintf(stderr, "%s <expr>\n", argv[0]);
|
||
|
return 0;
|
||
|
}
|
||
|
}
|