我正在使用flex和yacc移动一个小型线程解释器到re2c和lemon。一切正常,除了文字。
为什么与字面值相关的操作不能像运行yacc那样运行?我期望"1.0 end",但得到"0.0 end"
dspgrammar.y
%include {#include <assert.h>}
%name dsp
%token_type {float}
program ::= expr. {printf("endn");}
expr(val) ::= LITERAL. {printf("%f ", val);}
c
#include <stdio.h>
#include <stdlib.h>
#include "dspgrammar.h"
void *dspAlloc(void *(*)(size_t));
void dsp(void *, int, float);
void dspFree(void *, void (*)(void *));
void dspTrace(FILE *, char *);
int main(int argc, char *argv[])
{
void *parser = dspAlloc(malloc);
dspTrace(stderr, "TRACE: ");
dsp(parser, LITERAL, 1.0f);
dsp(parser, 0, 0.0f);
dspFree(parser, free);
return EXIT_SUCCESS;
}
Makefile
CC = gcc
CFLAGS = -O0 -g -Wall -Wextra -pedantic -std=gnu99
LD = gcc
LDFLAGS = -lm
dsp: main.o dspgrammar.o
$(LD) $(CFLAGS) -o $@ $^ $(LDFLAGS)
main.o: main.c
$(CC) $(CFLAGS) -c main.c
dspgrammar.o: dspgrammar.c
$(CC) -c $(CFLAGS) -c dspgrammar.c
dspgrammar.c: dspgrammar.y
lemon dspgrammar.y
In
expr(val) ::= LITERAL. { /* something with val */ }
val
为还原值名称。也就是说,它对应于yacc中的$$
。终端LITERAL
的语义值是$1
,因为您没有为该值提供符号名称。
也许你的意思是:
expr ::= LITERAL(val). { /* something with val */ }
或
expr(e) ::= LITERAL(v). { e = v; /* some other action */ }