- Published on
Bison/Flex で簡単な計算機を作成する
- Authors
- Name
- Eito YONEYAMA
- @ynym_8
本記事では, コンパイラコンパイラを使用して簡単なREPL風計算機を作成する. なお, REPL風とは, 見た目がREPL風という意味であり, REPLの機能を実装するわけではない. また, 本記事で使用している計算機環境は以下の通りである.
計算機環境
OS: Ubuntu 22.04.4 LTS
Kernel: Linux 5.15.153.1-microsoft-standard-WSL2, x86-64 architecture
Virtualization: WSL2
Bison
Bison とは, 構文解析器を生成するパーサジェネレータの一種であり, Yacc (Yet another compiler compiler) の上位互換ともいえる.
Flex
Flex とは, 字句解析器を生成するレキシカルアナライザの一種であり, Lex (Lexical analyser generator) の上位互換ともいえる.
Yacc/Lex (Bison/Flex) の記述方法については以下を参照されたい.
- GNU Bison - The Yacc-compatible Parser Generator
- westes/flex
- Introducing Flex and Bison
- Better error handling using Flex and Bison
実装
Lexer
まずは, 字句解析器を作成する.
calc.l
%{
#include <stdlib.h>
#include <stdio.h>
#include "calc.tab.h"
extern YYSTYPE yylval;
void yyerror(char*);
%}
%%
"+" { return OP_ADD; }
"-" { return OP_SUB; }
"*" { return OP_MUL; }
"/" { return OP_DIV; }
"^" { return OP_POW; }
"(" { return SYM_PRNL; }
")" { return SYM_PRNR; }
"," { return SYM_COMMA; }
"π"|"pi"|"PI" { return SYM_PI; }
"e"|"E" { return SYM_E; }
"sin"|"SIN" { return FUNC_SIN; }
"cos"|"COS" { return FUNC_COS; }
"tan"|"TAN" { return FUNC_TAN; }
"cot"|"COT" { return FUNC_COT; }
"sec"|"SEC" { return FUNC_SEC; }
"csc"|"CSC" { return FUNC_CSC; }
"sqrt"|"SQRT" { return FUNC_SQRT; }
"ceil"|"CEIL" { return FUNC_CEIL; }
"floor"|"FLOOR" { return FUNC_FLOOR; }
"abs"|"fabs"|"ABS"|"FABS" { return FUNC_ABS; }
"exp"|"EXP" { return FUNC_EXP; }
"log"|"LOG" { return FUNC_LOG; }
"log10"|"LOG10" { return FUNC_LOG10; }
"log7"|"LOG7" { return FUNC_LOG7; }
"log5"|"LOG5" { return FUNC_LOG5; }
"log3"|"LOG3" { return FUNC_LOG3; }
"log2"|"LOG2" { return FUNC_LOG2; }
"ln"|"LN" { return FUNC_LN; }
[0-9]+(\.[0-9]+)? {
sscanf(yytext, "%lf", &yylval);
return NUM;
}
\n { return EOL; }
[ \t]+ {}
"quit"|"exit" { return CMD_EXT; }
[a-zA-Z]+ {
yyerror("ERROR: Unrecognized input!");
}
. {
yyerror("ERROR: Unrecognized input!");
}
%%
int yywrap() {
return 1;
}
Parser
次に, 構文解析器を作成する.
calc.y
%{
#include <stdio.h>
#include <stdlib.h>
#include <math.h>
#define YYSTYPE double
int yylex(void);
void yyerror(char*);
%}
%token NUM
%token EOL
%token SYM_PRNL SYM_PRNR SYM_COMMA SYM_PI SYM_E
%token FUNC_ABS FUNC_SIN FUNC_COS FUNC_TAN FUNC_COT FUNC_SEC FUNC_CSC FUNC_SQRT FUNC_CEIL FUNC_FLOOR
%token FUNC_LOG FUNC_LOG10 FUNC_LOG7 FUNC_LOG5 FUNC_LOG3 FUNC_LOG2 FUNC_LN FUNC_EXP
%token CMD_EXT
%left OP_ADD OP_SUB
%left OP_MUL OP_DIV
%left OP_POW
%%
g : g e EOL { printf("%lf\n\nSimple-calculator> ", $2); }
| g EOL { printf("Simple-calculator> "); }
| g CMD_EXT { printf("Bye!\n\n"); YYACCEPT; }
|
;
e : NUM
| SYM_PI { $$ = M_PI; }
| SYM_E { $$ = M_E; }
| e OP_ADD e { $$ = $1 + $3; }
| e OP_SUB e { $$ = $1 - $3; }
| e OP_MUL e { $$ = $1 * $3; }
| e OP_DIV e {
if ($3 == 0) {
yyerror("ERROR: Division by zero!");
YYABORT;
} else {
$$ = $1 / $3;
}
}
| e OP_POW e { $$ = pow($1, $3); }
| SYM_PRNL e SYM_PRNR { $$ = $2; }
| FUNC_SQRT SYM_PRNL e SYM_PRNR { $$ = sqrt($3); }
| FUNC_CEIL SYM_PRNL e SYM_PRNR { $$ = ceil($3); }
| FUNC_FLOOR SYM_PRNL e SYM_PRNR { $$ = floor($3); }
| FUNC_ABS SYM_PRNL e SYM_PRNR { $$ = fabs($3); }
| FUNC_EXP SYM_PRNL e SYM_PRNR { $$ = exp($3); }
| FUNC_SIN SYM_PRNL e SYM_PRNR { $$ = sin($3); }
| FUNC_COS SYM_PRNL e SYM_PRNR { $$ = cos($3); }
| FUNC_TAN SYM_PRNL e SYM_PRNR { $$ = sin($3) / cos($3); }
| FUNC_COT SYM_PRNL e SYM_PRNR { $$ = cos($3) / sin($3); }
| FUNC_SEC SYM_PRNL e SYM_PRNR { $$ = 1 / cos($3); }
| FUNC_CSC SYM_PRNL e SYM_PRNR { $$ = 1 / sin($3); }
| FUNC_LOG10 SYM_PRNL e SYM_PRNR { $$ = log($3) / log(10); }
| FUNC_LOG7 SYM_PRNL e SYM_PRNR { $$ = log($3) / log(7); }
| FUNC_LOG5 SYM_PRNL e SYM_PRNR { $$ = log($3) / log(5); }
| FUNC_LOG3 SYM_PRNL e SYM_PRNR { $$ = log($3) / log(3); }
| FUNC_LOG2 SYM_PRNL e SYM_PRNR { $$ = log($3) / log(2); }
| FUNC_LN SYM_PRNL e SYM_PRNR { $$ = log($3); }
| FUNC_LOG SYM_PRNL e SYM_COMMA e SYM_PRNR { $$ = log($5) / log($3); }
| OP_ADD e { $$ = $2; }
| OP_SUB e { $$ = 0 - $2; }
| error { YYABORT; }
;
%%
void yyerror(char *s)
{
fprintf(stderr, "%s\n\n", s);
}
int main()
{
printf("Simple-calculator> ");
yyparse();
return 0;
}
コンパイル
実際にコンパイルしてみる. コンパイルは以下のように行う.
$ bison -d calc.y
$ flex calc.l
$ gcc -O2 -o calc calc.tab.c lex.yy.c -lfl -lm
calc.l: In function ‘yylex’:
calc.l:40:20: warning: format ‘%lf’ expects argument of type ‘double *’, but argument 3 has type ‘YYSTYPE *’ {aka ‘int *’} [-Wformat=]
40 | sscanf(yytext, "%lf", &yylval);
| ^~~~~ ~~~~~~~
| |
| YYSTYPE * {aka int *}
上記のコンパイルにより, 実行ファイルが生成される. なお, Warning は無視する. また, コンパイルの際には以下のような Makefile
を使用すると便利.
Makefile
all: calc
calc: calc.tab.c lex.yy.c
gcc -O2 -o calc calc.tab.c lex.yy.c -lfl -lm
calc.tab.c calc.tab.h: calc.y
bison -d calc.y
lex.yy.c: calc.l
flex calc.l
clean:
rm -f calc *.o *.c *.h y.tab.* lex.yy.c
Makefile の実行例
$ make
bison -d calc.y
flex calc.l
gcc -O2 -o calc calc.tab.c lex.yy.c -lfl -lm
calc.l: In function ‘yylex’:
calc.l:40:20: warning: format ‘%lf’ expects argument of type ‘double *’, but argument 3 has type ‘YYSTYPE *’ {aka ‘int *’} [-Wformat=]
40 | sscanf(yytext, "%lf", &yylval);
| ^~~~~ ~~~~~~~
| |
| YYSTYPE * {aka int *}
使用例
四則演算
$ ./calc
Simple-calculator> 1 + 2
3.000000
Simple-calculator> 1 - 2
-1.000000
Simple-calculator> 1 / 2
0.500000
Simple-calculator> 2 * 3
6.000000
三角関数
$ ./calc
Simple-calculator> sin(pi/2)
1.000000
Simple-calculator> cos(pi)
-1.000000
Simple-calculator> tan(0)
0.000000
Simple-calculator> sec(pi/4)
1.414214
Simple-calculator> cot(pi/4)
1.000000
対数関数
$ ./calc
Simple-calculator> ln(e)
1.000000
Simple-calculator> log10(10)
1.000000
Simple-calculator> log2(4)
2.000000
その他関数
$ ./calc
Simple-calculator> abs(-1)
1.000000
Simple-calculator> floor(1.9)
1.000000
Simple-calculator> ceil(2.1)
3.000000
Simple-calculator> sqrt(4)
2.000000
Simple-calculator> exp(2)
7.389056
終了コマンド
$ ./calc
Simple-calculator> exit
Bye!
$ ./calc
Simple-calculator> quit
Bye!
エラー処理
未定義エラー
$ ./calc
Simple-calculator> exi
ERROR: Unrecognized input!
ゼロ除算エラー
$ ./calc
Simple-calculator> 1 / 0
ERROR: Division by zero!
構文エラー
$ ./calc
Simple-calculator> abs
syntax error
まとめ
今回は, Bison/Flexを使用して簡単な計算機を作成したが, 機能面やエラー処理等において不十分な点が多いため, 時間があれば改善したい.