aboutsummaryrefslogtreecommitdiff
path: root/src/sharing.c
blob: 3219860a2d9674dc9ad4b7856f4ba9189e53bca9 (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
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
// Copyright (c) 2023, Marvin Borner <dev@marvinborner.de>
// SPDX-License-Identifier: MIT

#include <stdlib.h>

#include <lib/list.h>
#include <log.h>
#include <sharing.h>
#include <map.h>

// Procedure EnqueueAndPropagate()
static void enqueue_and_propagate(struct term *neighbour, struct term *term)
{
	if (neighbour->type == ABS && term->type == ABS) {
		// m' ~ c'
		neighbour->u.abs.term->neighbours = list_add(
			neighbour->u.abs.term->neighbours, term->u.abs.term);
		term->u.abs.term->neighbours = list_add(
			term->u.abs.term->neighbours, neighbour->u.abs.term);
	} else if (neighbour->type == APP && term->type == APP) {
		// m1 ~ c1
		neighbour->u.app.lhs->neighbours = list_add(
			neighbour->u.app.lhs->neighbours, term->u.app.lhs);
		term->u.app.lhs->neighbours = list_add(
			term->u.app.lhs->neighbours, neighbour->u.app.lhs);

		// m2 ~ c2
		neighbour->u.app.rhs->neighbours = list_add(
			neighbour->u.app.rhs->neighbours, term->u.app.rhs);
		term->u.app.rhs->neighbours = list_add(
			term->u.app.rhs->neighbours, neighbour->u.app.rhs);
	} else if (neighbour->type == VAR && term->type == VAR) {
	} else {
		fatal("query failed\n");
	}
	neighbour->canonic = term;
	queue_push(term->queue, neighbour);
}

// Procedure BuildEquivalenceClass()
static void build_equivalence_class(struct term *term)
{
	term->canonic = term;
	term->building = 1;
	term->queue = queue_new();
	queue_push(term->queue, term);

	while (!queue_empty(term->queue)) {
		struct term *cur = queue_pop(term->queue);

		struct list *iterator = cur->parents;
		while (iterator) {
			struct term *parent = iterator->data;
			if (!parent->canonic) {
				build_equivalence_class(parent);
			} else if (parent->canonic->building) {
				fatal("query failed\n");
			}
			iterator = iterator->next;
		}

		iterator = cur->neighbours;
		while (iterator) {
			struct term *neighbour = iterator->data;
			if (!neighbour->canonic) {
				enqueue_and_propagate(neighbour, term);
			} else if (neighbour->canonic != term) {
				fatal("query failed\n");
			}
		}
	}

	term->building = 0;
}

static void blind_check_callback(struct term *term)
{
	if (term->canonic)
		return;
	build_equivalence_class(term);
}

// Procedure BlindCheck()
static void blind_check(void)
{
	map_foreach(blind_check_callback);
}

static void var_check_callback(struct term *term)
{
	if (term->type != VAR)
		return;
	struct term *canonic = term->canonic;
	if (canonic != term) {
		// you should compare the binders as stated in the paper.
		// due to the implementation I just compare the indices directly
		if (canonic->u.var.index != term->u.var.index) {
			fatal("query failed\n");
		}
	}
}

// Procedure VarCheck()
static void var_check(void)
{
	map_foreach(var_check_callback);
}

// this doesn't really make sense for this specific implementation of
// λ-graphs using hashes. It's a reference implementation after all.
void sharing_query(struct term *a, struct term *b)
{
	a->neighbours = list_add(a->neighbours, b);
	b->neighbours = list_add(b->neighbours, a);

	blind_check();
	var_check();
}