Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/triska/clpb

Boolean Constraint Solving in Prolog
https://github.com/triska/clpb

bdd clp constraint-programming constraints independent-sets matchsticks-puzzle prolog sat satisfiability

Last synced: about 2 months ago
JSON representation

Boolean Constraint Solving in Prolog

Awesome Lists containing this project

README

        

# CLP(B): Constraint Logic Programming over Boolean variables

CLP(B), Constraint Logic Programming over Boolean variables, is
available in Scryer Prolog and SWI-Prolog as
[**library(clpb)**](https://www.scryer.pl/clpb.html).

This repository contains usage examples and tests of the library.

[**clpb.pdf**](clpb.pdf) is a _shortened version_ of the library
documentation, intended as supplementary lecture material.

**Project page**:

[**https://www.metalevel.at/clpb/**](https://www.metalevel.at/clpb/)

## Using CLP(B) constraints

Many of the examples use
[**DCG notation**](https://www.metalevel.at/prolog/dcg) to
describe lists of clauses. This lets you easily reason about the
constraints that are being posted, change the order in which they are
posted, and in general more conveniently experiment with CLP(B).
In some examples, it is faster to post a single big conjunction
instead of several smaller ones.

I recommend you start with the following examples:

1. [**knights_and_knaves.pl**](knights_and_knaves.pl): Solution of
several Boolean puzzles that appear in Raymond Smullyan's _What Is
the Name of this Book_ and Maurice Kraitchik's _Mathematical
Recreations_. These examples and other
[**logic puzzles**](https://www.metalevel.at/prolog/puzzles)
are good starting points for learning more about CLP(B).

2. [**xor.pl**](xor.pl): Verification of a digital circuit, expressing
XOR with NAND gates:

![](figures/filler.png) ![XOR with NAND gates](figures/xor.png)

This example uses *universally quantified* variables to express the
output as a function of the input variables in residual goals.

3. [**matchsticks.pl**](matchsticks.pl): A puzzle involving
matchsticks. See below for more information.

4. [**cycle_n.pl**](cycle_n.pl): Uses Boolean constraints to express
independent sets and *maximal* independent sets (also called
*kernels*) of the
[cycle graph](https://en.wikipedia.org/wiki/Cycle_graph) CN.

![](figures/filler.png) ![Cycle graph C_7](figures/cycle7.png) ![](figures/filler20.png) ![Kernel of C_7](figures/cycle7_kernel.png)

See below for more information about weighted solutions.

5. [**euler_172.pl**](euler_172.pl): CLP(B) solution of Project Euler
[Problem 172](https://projecteuler.net/problem=172): How many
18-digit numbers n (without leading zeros) are there
such that no digit occurs more than three times in n?

6. [**domino_tiling.pl**](domino_tiling.pl): Domino tiling of an
M×N chessboard. Using CLP(B), it is easy to see
that there are 12,988,816 ways to cover an
8×8 chessboard with dominoes:

![](figures/filler.png) ![Domino tiling of an 8x8 chessboard](figures/domino8x8.png) ![](figures/filler20.png) ![Domino tiling of a 2x8 chessboard](figures/domino2x8.png)

Interestingly, the
[Fibonacci numbers](http://mathworld.wolfram.com/FibonacciNumber.html)
arise when we count the number of domino tilings of
2×N chessboards. An example is shown in the right
figure.

Other examples are useful as benchmarks:

- [**langford.pl**](langford.pl): Count the number of [Langford pairings](https://en.wikipedia.org/wiki/Langford_pairing).
- [**n_queens.pl**](n_queens.pl): CLP(B) formulation of the
[N-queens puzzle](https://en.wikipedia.org/wiki/Eight_queens_puzzle).
- [**pigeon.pl**](pigeon.pl): A simple allocation task.
- [**schur.pl**](schur.pl): A problem related to
[Schur's number](http://mathworld.wolfram.com/SchurNumber.html) as
known from
[Ramsey theory](http://mathworld.wolfram.com/RamseyTheory.html).

The directory [**bddcalc**](bddcalc) contains a very simple calculator
for BDDs.

### Matchsticks puzzle

In [**matchsticks.pl**](matchsticks.pl), Boolean variables indicate
whether a matchstick is placed at a specific position. The task is to
eliminate all subsquares from the initial configuration in such a way
that the maximum number of matchsticks is left in place:

![](figures/filler.png) ![Matchsticks initial configuration](figures/matchsticks1.png)

We can use the CLP(B) predicate `weighted_maximum/3` to show that we
need to remove at least 9 matchsticks to eliminate all subsquares.

![](figures/filler.png) ![Matchsticks without any subsquares](figures/matchsticks2.png) ![](figures/filler.png) ![Exactly 7 subsquares remaining](figures/matchsticks3.png)

The left figure shows a sample solution, leaving the maximum number of
matchsticks (31) in place. If you keep more matchsticks in place,
subsquares remain. For example, the right figure contains exactly
7 subsquares, including the 4x4 outer square.

CLP(B) constraints can be used to quickly generate, test and count
solutions of such puzzles, among many other applications. For example,
there are precisely 62,382,215,032 subsquare-free configurations that
use exactly 18 matchsticks. This is the maximum number of such
configurations for any fixed number of matchsticks on this grid.

### Independent sets and weighted kernels

As another example, consider the following graph:

![](figures/filler20.png) ![Cycle graph with 100 nodes, C_100](figures/cycle100.png)

It is the so-called
[_cycle graph_](https://en.wikipedia.org/wiki/Cycle_graph) with
100 nodes, C100. Using CLP(B) constraints, it
is easy to see that this graph has exactly 792,070,839,848,372,253,127
_independent sets_, and exactly 1,630,580,875,002 _maximal_
independent sets, which are also called _kernels_. The gray nodes in
the next picture show one such kernel:

![](figures/filler20.png) ![Maximal independent set of C_100](figures/cycle100_maximum.png)

Suppose that we assign each node nj the weight
wj =  (-1)νj, where νj
denotes the number of ones in the binary representation
of j. In the above figure, nodes with negative weight are
drawn as squares, and nodes with positive weight are drawn as circles.

Only 5 nodes (1, 25, 41, 73 and 97) of this kernel with 38 nodes have
negative weight in this case, for a total weight of 28. In this case,
the example shows a kernel with maximum weight. It is easy to
find such kernels with the CLP(B) predicate `weighted_maximum/3`, and
we can also compute other interesting facts: For example, there are
exactly 256 kernels of maximum weight in this case. There are exactly
25,446,195,000 kernels with exactly 38 nodes. All kernels have between
34 and 50 nodes. For any fixed number of nodes, the maximum number of
kernels (492,957,660,000) is attained with 41 nodes, and among these
kernels, the maximum total weight is 25.

By negating the coefficients of `weighted_maximum/3`, we can also find
kernels with _minimum_ weight. For example:

![](figures/filler20.png) ![Kernel of C_100 with minimum weight](figures/cycle100_minimum.png)

## Implementation

The implementation of `library(clpb)` is based on ordered and reduced
**Binary Decision Diagrams** (BDDs). BDDs are an important data
structure for representing Boolean functions and have many virtues
that often allow us to solve interesting tasks efficiently.

For example, the CLP(B) constraint `sat(card([2],[X,Y,Z]))` is
translated to the following BDD:

![](figures/filler20.png) ![BDD for sat(card([2],[X,Y,Z]))](http://www.metalevel.at/card.svg)

To inspect the BDD representation of Boolean constraints, set the
Prolog flag `clpb_residuals` to `bdd`. For example:

?- set_prolog_flag(clpb_residuals, bdd).
true.

?- sat(X+Y).
node(2)- (v(X, 0)->true;node(1)),
node(1)- (v(Y, 1)->true;false).

Using `library(clpb)` is a good way to learn more about BDDs. The
variable order is determined by the order in which the variables are
first encountered in constraints. You can enforce arbitrary variable
orders by first posting a tautology such as `+[1,V1,V2,...,Vn]`.

For example:


?- sat(+[1,Y,X]), sat(X+Y).
node(2)- (v(Y, 0)->true;node(1)),
node(1)- (v(X, 1)->true;false).

You can render CLP(B)'s residual goals as BDDs in SWISH using the
[**BDD renderer**](http://swish.swi-prolog.org/example/render_bdd.swinb).

## ZDD-based variant of `library(clpb)`

There is a limited alternative version of `library(clpb)`, based on
Zero-suppressed Binary Decision Diagrams (ZDDs).

Please see the [**zdd**](zdd) directory for more information. Try the
ZDD-based version for tasks where the BDD-based version runs out of
memory. You must use `zdd_set_vars/1` before using `sat/1` though.

## Acknowledgments

I am extremely grateful to:

[**Jan Wielemaker**](http://eu.swi-prolog.org) for providing the
Prolog system that made all this possible in the first place.

[**Ulrich Neumerkel**](http://www.complang.tuwien.ac.at/ulrich/), who
introduced me to constraint logic programming and CLP(B) in
particular. If you are teaching Prolog, I strongly recommend you check
out his
[GUPU system](http://www.complang.tuwien.ac.at/ulrich/gupu/).

[**Nysret Musliu**](http://dbai.tuwien.ac.at/staff/musliu/), my thesis
advisor, whose interest in combinatorial tasks, constraint
satisfaction and SAT solving highly motivated me to work in this
area.

[**Mats Carlsson**](https://www.sics.se/~matsc/), the designer and
main implementor of SICStus Prolog and its visionary
[CLP(B) library](https://sicstus.sics.se/sicstus/docs/latest4/html/sicstus.html/lib_002dclpb.html#lib_002dclpb).
For any serious use of Prolog and constraints, make sure to check out
his elegant and fast system.

[**Donald Knuth**](http://www-cs-faculty.stanford.edu/~uno/) for the
superb treatment of BDDs and ZDDs in his books and programs.