aboutsummaryrefslogtreecommitdiff
path: root/src/lyn.c
blob: 252767756d984db91665b5eb3c8ff4e1077ca663 (plain) (blame)
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
79
80
81
82
83
84
85
86
87
88
89
#include <errno.h>
#include <stdbool.h>
#include <string.h>
#include <stdio.h>
#include <limits.h>
#include <stdlib.h>

#include <lyn/lyn.h>
#include <lyn/parser.h>
#include <lyn/debug.h>

struct lyn lyn_create()
{
	return (struct lyn){};
}

int lyn_eval_str(struct lyn *lyn, const char *name, const char *str)
{
	struct parser *p = create_parser();
	if (!p)
		return -1;

	parse(p, name, str);
	struct ast *ast = p->tree;
	bool failed = p->failed;
	destroy_parser(p);

	if (!failed) {
		ast_dump_list(0, ast);
	}

	return failed;
}

/**
 * Read whole file into a buffer and return pointer to buffer.
 * Possibly kind of silly to have both \p file and \p f.
 * Apparently there's no standardized way to get the file name of a
 * file pointer.
 *
 * @param file Name of file to read.
 * @param f File pointer.
 * @return Pointer to buffer with file contents.
 */
static char *read_file(const char *file, FILE *f)
{
	fseek(f, 0, SEEK_END);
	/** @todo check how well standardized this actually is */
	long s = ftell(f);
	if (s == LONG_MAX) {
		error("%s might be a directory", file);
		return NULL;
	}

	fseek(f, 0, SEEK_SET);

	char *buf = malloc((size_t)(s + 1));
	if (!buf)
		return NULL;

	fread(buf, (size_t)(s + 1), 1, f);
	/* remember terminating null */
	buf[s] = 0;
	return buf;
}

int lyn_eval_file(struct lyn *lyn, const char *fname)
{
	FILE *f = fopen(fname, "rb");
	if (!f) {
		error("failed opening %s: %s\n", fname, strerror(errno));
		return -1;
	}

	char *buf = read_file(fname, f);
	fclose(f);

	if (!buf)
		return -1;

	int ret = lyn_eval_str(lyn, fname, buf);
	free(buf);

	return ret;
}

void lyn_destroy(struct lyn *lyn)
{
}