summaryrefslogtreecommitdiff
path: root/src/runtime/swipl/swipgf.c
blob: 39b683217c3f97898e5765daf6da44537893d71d (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
79
80
81
82
#include <SWI-Prolog.h>
#include <pgf/pgf.h>

typedef struct {
	GuPool* pool;
	PgfPGF* pgf;
} PlPGF;

static PL_blob_t pgf_blob = {
	PL_BLOB_MAGIC,
	PL_BLOB_UNIQUE,
	"PGF",
	NULL,
	NULL,
	NULL,
	NULL
};

static foreign_t
pl_readPGF(term_t a1, term_t a2)
{
	char *fpath;

	if (!PL_get_atom_chars(a1, &fpath) )
		PL_fail;
		
	PlPGF pl_pgf;
	pl_pgf.pool = gu_new_pool();

	GuPool* tmp_pool = gu_local_pool();

	// Create an exception frame that catches all errors.
	GuExn* err = gu_new_exn(NULL, gu_kind(type), tmp_pool);

	// Read the PGF grammar.
	pl_pgf.pgf = pgf_read(fpath, pl_pgf.pool, err);
	if (!gu_ok(err)) {
		int res;
		if (gu_exn_caught(err) == gu_type(GuErrno)) {
			errno = *((GuErrno*) gu_exn_caught_data(err));
			res = PL_existence_error("source_sink", a1);
		} else {
			res = PL_raise_exception(a1);
		}
		gu_pool_free(pl_pgf.pool);
		gu_pool_free(tmp_pool);
		return res;
	}

	gu_pool_free(tmp_pool);

	if (!PL_unify_blob(a2, &pl_pgf, sizeof(pl_pgf), &pgf_blob)) {
		gu_pool_free(pl_pgf.pool);
		PL_fail;
	}

	PL_succeed;
}

static foreign_t
pl_language(term_t a1, term_t a2, term_t a3, control_t handle)
{
	switch (PL_foreign_control(handle)) {
	case PL_FIRST_CALL:
        //ctxt = malloc(sizeof(struct context));
        PL_retry_address(NULL/*ctxt*/);
    case PL_REDO:
        //ctxt = PL_foreign_context_address(handle);
        PL_retry_address(NULL/*ctxt*/);
    case PL_PRUNED:
        //ctxt = PL_foreign_context_address(handle);
        //free(ctxt);
        PL_succeed;
	}
	PL_succeed;
}

install_t
install_swipgf()
{ PL_register_foreign("readPGF", 2, pl_readPGF, 0);
  PL_register_foreign("language", 3, pl_language, PL_FA_NONDETERMINISTIC);
}