forked from Jianchu/generic-type-inference-solver
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathlogiqlEncoding.tex
131 lines (113 loc) · 8.01 KB
/
logiqlEncoding.tex
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
119
120
121
122
123
124
125
126
127
128
129
130
131
\documentclass[11pt]{article}
\usepackage[utf8]{inputenc}
\usepackage[english]{babel}
\usepackage{slashbox}
\usepackage{listings}
\lstset{
breaklines=true,
}
\usepackage{indentfirst}
\usepackage{natbib}
\usepackage{graphicx}
\usepackage{amsmath}
\begin{document}
\section{Encoding}
\par Given a type system with $n$ qualifiers, and a slot whose ID is $slotId$. Then we can use an integer value range from $((slotId - 1) * n + 1)$ to $((slotId - 1) * n + n)$ represent each type qualifier of that slot.
\par For example, if we consider about the universe type system, the representation is showing in following table:
\begin{center}
\begin{tabular}{|l|l|l|l|l|}\hline
\backslashbox{$SlotId$}{$Qualifier$} & peer & rep & lost & any\\
\hline
1 & 1 & 2 & 3 & 4\\
\hline
2 & 5 & 6 & 7 & 8\\
\hline
3 & 9 & 10 & 11 & 12\\
\hline
... & & & & \\
\hline
9 & 33 & 34 & 35 & 36\\
\hline
... & & & &\\
\hline
12 & 45 & 46 & 47 & 48\\
\hline
\end{tabular}
\end{center}
\par In the above table, every number in the middle represents the corresponding slot is annotated corresponding qualifier. We call those numbers \textit{variables}. For example $6$ means slot 2 is annotated qualifier rep.
\par Now, we will combine this table with the constraints generated from \textit{Checker Framework Inference}. We noticed that every constraint could be encoded as an "implies relation". For example, if we get a \textit{Equality Constraint: $1 = 2$}, where 1 and 2 are both slot ID. Then this constraint can be encoded into $(1 \implies 5) \wedge (2 \implies 6) \wedge (3 \implies 7) \wedge (4 \implies 8)$.
\par Based on the above observation, we abstracted all the type constraints into the a bunch of implies logic. Then encode the those implies logic as MAX-SAT problem and LogiQL language, and solve them by sat solver and LogicBlox.
\subsection{SAT encoding}
\subsection{LogiQL encoding}
\par LogiQL is a declarative logic programming language ran by LogicBlox database system. Given a set of constraints from a particular type system, we can encode those constraints into LogiQL form, and solve them by LogicBlox.
\subsubsection{Basic Encoding}
Basic encoding consists some predicates that won't depend on type systems.
\begin{lstlisting}
variable(v),hasvariableName(v:i)->int(i).
\end{lstlisting}
\par This predicate stores all the variables, which will be used for all other tables.
\begin{lstlisting}
isAnnotated[v] = i -> variable(v), boolean(i).
\end{lstlisting}
\par This predicate is a functional predicate mapping variable entities to boolean values. Boolean value equals to true means the corresponding slot is annotated by corresponding annotation.
%If the value equals to false, then the slot mustn't be annotated by corresponding annotation, this may happen if \textit{Inequality Constraint} or \textit{Comparable Constraint} show in the constraints set.
\begin{lstlisting}
hasToBeTrue[v] = h -> variable(v), boolean(h).
\end{lstlisting}
\par This predicate is for the situation that a variable must be true, this predicate is only used for the case that there is an equality constraint between a constant slot and a variable slot, a variable slot is supertype of the top type, or a variable slot is subtype of the bottom type.
\begin{lstlisting}
set_1_variable[v] = s -> variable(v), boolean(s).
\end{lstlisting}
\par The predicate's name has the pattern "set\_x\_variable" means a group of variable, x is the number of variables in this set, the whole predicate would be true if one of variables in this set is true.
\begin{lstlisting}
set_1_variable[v] = true <- hasToBeTrue[v] = true.
\end{lstlisting}
\par \textit{hasToBeTrue[v] = true} is equivalent to \textit{set\_1\_variable[v] = true}, because both of them mean variable v is true. And in other steps, the information in \textit{set\_1\_variable[v]} will be used, so that we make this rule ensuring textit{set\_1\_variable[v]} contains all necessary information.
\begin{lstlisting}
implies1[v1,v2] = e -> variable(v1), variable(v2), boolean(e).
\end{lstlisting}
\par The predicate that has the pattern "impliesx" represents the "implies" logic in our encoding. "x" indicates how many variables in the right hand side of implies logic. For example, implies1[1,3] = true, means "$1\implies3$".
\begin{lstlisting}
rightSide[v] = r -> variable(v), boolean(r).
\end{lstlisting}
\par rightSide predicate indicates whether the right side of logic implies operated by logic negation. For example, implies1[1, 2] = true, rightSide[2] = false means $1 \implies\neg3$.
\begin{lstlisting}
set_1_variable[v2] = true <- implies[v1,v2] = true, set_1_variable[v1] = true, rightSide[v2] = true.
set_1_variable[v2] = false <- implies[v1,v2] = true, set_1_variable[v1] = true, rightSide[v2] = false.
\end{lstlisting}
\par These two rules indicate that the implies relationship between two variables, the value of v2 in set\_1\_variable depends on the value of v1 , and whether there is a negation in right side of implies. The first one could be read as "If v1 implies v2, v1 is true, and there is no negation for the right side of implies logic, then v2 is true."
\par These two rules are basic encoding because every type systems has at least one type qualifier.
\begin{lstlisting}
isAnnotated[v] = true <- set_1_variable[v] = true.
isAnnotated[v] = false <- set_1_variable[v] = false.
\end{lstlisting}
\par Predicate \textit{isAnnotated[v]} is the place that we store all the results. \textit{isAnnotated[v] = true} means the slot that represented by v is annotated by corresponding qualifier, and \textit{isAnnotated[v] = false} is the opposed case.
\begin{lstlisting}
implies2[v1,v2,v3] = e -> variable(v1), variable(v2), variable(v3),boolean(e).
set_2_variable[v1,v2] = e ->variable(v1),variable(v2),boolean(e).
\end{lstlisting}
\par The idea of predicate \textit{implies2[v1,v2,v3]} is same as the \textit{implies1[v1,v2]}, but it represents the implies relation among three variables.
\par Similarly the predicate \textit{set\_2\_variable[v1,v2]} is used for 2 variables.
\begin{lstlisting}
set_2_variable[v2,v3] = true <- implies2[v1,v2,v3] = true, set_1_variable[v1] = true;implies2[v1,v2,v3] = true,isAnnotated[v1] = true.
\end{lstlisting}
\par The above rule can be read as "If $v1\implies (v2 \vee v3)$, and $v1$ is true, then one of $v2$ and $v3$ has to be true. We put union two conditions here because both \textit{isAnnotated} and \textit{set\_1\_variable} can indicate the truth value of one variable."
\begin{lstlisting}
isAnnotated[v1] = true <- set_2_variable[v1,_] = true, !set_1_variable[v1] = false.
isAnnotated[v2] = true <- set_2_variable[v1,v2] = true, isAnnotated[v1] = false,!set_1_variable[v2] = false.
\end{lstlisting}
\par If we know the fact that one of v1 and v2 is true, then we will check v1 and v2 one by one, if there is no constraint says v1 cannot be false, we set \textit{isAnnotated[v1] = true}. If v1 mustn't be true, then we will check whether v2 could be true.
\par All predicates listed above are general encoding because every type system needs them, but if we only have the above predicates, we can only solve the constraint generated for the type systems containing only 2 qualifiers. In order to handle the type system has more qualifiers, some predicate like
\begin{lstlisting}
implies3[v1,v2,v3,v4] = e -> variable(v1), variable(v2), variable(v3), variable(v4),boolean(e).
set_3_variable[v1,v2,v3] = e ->variable(v1),variable(v2),variable(v3),boolean(e).
set_3_variable[v2,v3,v4] = true <- implies3[v1,v2,v3,v4] = true, set_1_variable[v1] = true;implies3[v1,v2,v3,v4] = true,isAnnotated[v1] = true.
\end{lstlisting}
need to be introduced. And we need to following rules as well:
\begin{lstlisting}
implies3[v1,v2,v3,v4] = e -> variable(v1), variable(v2), variable(v3), variable(v4),boolean(e).
isAnnotated[v1] = true <- set_3_variable[v1,_,_] = true, !set_1_variable[v1] = false.
isAnnotated[v2] = true <- set_3_variable[v1,v2,_] = true, isAnnotated[v1] = false,!set_1_variable[v2] = false.
isAnnotated[v3] = true <- set_3_variable[v1,v2,v3] = true, isAnnotated[v1] = false,isAnnotated[v2] = false,!set_1_variable[v3] = false.
\end{lstlisting}
\end{document}