-
-
Save rui314/a13acb5bbea0fb4bd66eaa4db4ac93d2 to your computer and use it in GitHub Desktop.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
#include <stdio.h> | |
#include <ctype.h> | |
#include <stdarg.h> | |
#include <stdlib.h> | |
#include <string.h> | |
static char *p; | |
static char func[26][100]; | |
__attribute__((noreturn)) static void error(char *fmt, ...) { | |
va_list ap; | |
va_start(ap, fmt); | |
vfprintf(stderr, fmt, ap); | |
fprintf(stderr, "\n"); | |
exit(1); | |
} | |
static void read_until(char c, char *buf) { | |
for (; *p != c; p++, buf++) | |
*buf = *p; | |
p++; | |
*buf = '\0'; | |
} | |
static void skip() { | |
while (isspace(*p)) | |
p++; | |
} | |
static void expect(char c) { | |
if (*p != c) | |
error("%c expected: %s", c, p); | |
p++; | |
} | |
static int eval(int *args); | |
static int eval_string(char *code, int *args) { | |
char *orig = p; | |
p = code; | |
int val; | |
while (*p) | |
val = eval(args); | |
p = orig; | |
return val; | |
} | |
static int eval(int *args) { | |
skip(); | |
// Function parameter | |
if ('a' <= *p && *p <= 'z') | |
return args[*p++ - 'a']; | |
// Function definition | |
if ('A' <= *p && *p <= 'Z' && p[1] == '[') { | |
char name = *p; | |
p += 2; | |
read_until(']', func[name - 'A']); | |
return eval(args); | |
} | |
// "P" print primitive | |
if (*p == 'P') { | |
p++; | |
expect('('); | |
int val = eval(args); | |
expect(')'); | |
printf("%d\n", val); | |
return val; | |
} | |
// Function application | |
if ('A' <= *p && *p <= 'Z' && p[1] == '(') { | |
int newargs[26]; | |
char name = *p; | |
p += 2; | |
int i = 0; | |
for (skip(); *p != ')'; skip()) | |
newargs[i++] = eval(args); | |
expect(')'); | |
return eval_string(func[name - 'A'], newargs); | |
} | |
// Literal numbers | |
if (isdigit(*p)) { | |
int val = *p++ - '0'; | |
while (isdigit(*p)) | |
val = val * 10 + (*p++ - '0'); | |
return val; | |
} | |
// Arithmetic operators | |
if (strchr("+-*/", *p)) { | |
int op = *p++; | |
int x = eval(args); | |
int y = eval(args); | |
switch (op) { | |
case '+': return x + y; | |
case '-': return x - y; | |
case '*': return x * y; | |
case '/': return x / y; | |
} | |
} | |
error("invalid character: %c", *p); | |
} | |
int main(int argc, char **argv) { | |
p = argv[1]; | |
while (*p) | |
printf("%d\n", eval(0)); | |
return 0; | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment