+ print_token_kind(f, (token_kind_t)token->kind);
+ fprintf(f, " \"%s\"", token->string.string.begin);
+ break;
+ case T_CHARACTER_CONSTANT:
+ case T_WIDE_CHARACTER_CONSTANT:
+ print_token_kind(f, (token_kind_t)token->kind);
+ fputs(" \'", f);
+ print_stringrep(&token->string.string, f);
+ fputs("'", f);
+ break;
+ default:
+ fputc('\'', f);
+ print_token_kind(f, (token_kind_t)token->kind);
+ fputc('\'', f);
+ break;
+ }
+}
+
+void print_pp_token_kind(FILE *f, int token_kind)
+{
+ if (token_kind == TP_EOF) {
+ fputs("end of file", f);
+ return;
+ }
+ if (token_kind == TP_ERROR) {
+ fputs("error", f);
+ return;
+ }
+
+ int token_symbols_len = TP_LAST_TOKEN;
+ if (token_kind < 0 || token_kind >= token_symbols_len) {
+ fputs("invalid token", f);
+ return;
+ }
+
+ const symbol_t *symbol = pp_token_symbols[token_kind];
+ if (symbol != NULL) {
+ fputs(symbol->string, f);
+ } else {
+ if(token_kind >= 0 && token_kind < 256) {
+ fputc(token_kind, f);
+ return;
+ }
+ fputs("unknown token", f);
+ }
+}
+
+void print_pp_token(FILE *f, const token_t *token)
+{
+ switch((preprocessor_token_kind_t) token->kind) {
+ case TP_IDENTIFIER:
+ fprintf(f, "identifier '%s'", token->identifier.symbol->string);
+ break;
+ case TP_NUMBER:
+ fprintf(f, "number '%s'", token->number.number.begin);
+ break;
+ case TP_STRING_LITERAL:
+ fprintf(f, "string \"%s\"", token->string.string.begin);