summaryrefslogtreecommitdiff
path: root/tok_util.c
blob: c17ca6d12faff890904b8fdc0d9dd200e4117dad (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
#include <string.h>
#include <ctype.h>

#include "lex.h"

const char *tok_to_desc[] = {
	[TOKEN_REGISTER] = "register",
	[TOKEN_NUMERIC] = "numeric literal",
	[TOKEN_KEYWORD] = "keyword",
	[TOKEN_STRING] = "string literal",
	[TOKEN_COMMA] = "comma",
	[TOKEN_LABEL] = "label",
	[TOKEN_IDENT] = "identifier",
	[TOKEN_DOT] = "assembler directive",
	[TOKEN_EOL] = "end of line",
};

const char * get_token_description(enum TOKEN_TYPE t)
{
	if (t < 0 || t >= sizeof(tok_to_desc)/sizeof(*tok_to_desc)) {
		return "[internal error]";
	} else {
		return tok_to_desc[t];
	}
}

void indicate_file_area(FILE* fd, size_t line, size_t column, size_t span)
{
	size_t i = 0;
	const char margin[] = "  ";

	char buf[1024] = { '\0' };
	char *s = buf;
	char c = '\0';

	rewind(fd);
	while (line && !feof(fd) && fgets(buf, sizeof(buf), fd)) {
		s = buf;
		while (*s) {
			if (*(s++) == '\n') {
				line--;
			}
		}
	}

	/* trim leading whitespace */
	s = buf;
	while (*s == '\t' || *s == ' ') {
		s++;
	}

	/* filter non-printables to spaces to keep alignment correct */
	for (i = 0; i < strlen(s); i++) {
		if (!isprint(s[i]) && s[i] != '\n') {
			s[i] = ' ';
		}
	}

	fputs(margin, stderr);
	fputs(s, stderr);

	/* corner case (still needed?) - buf was just return */
	if (strlen(buf) == 1 && buf[0] == '\n') {
		fputc('\n', stderr);
	}

	fputs(margin, stderr);
	column -= (s - buf);
	for (column--; column; column--) {
		fputc(' ', stderr);
	}

	c = span == 1 ? '^' : '"';
	for (; span; span--) {
		fputc(c, stderr);
	}
	fputc('\n', stderr);
}