
search(+L, ++Arg, ++Select, +Choice, ++Method, +Option)

   A generic search routine for finite domains or IC which implements 
different partial search methods (complete, credit, lds, bbs, dbs)

Arguments
   L                   is a list of domain variables (Arg = 0) or a list of
	    terms (Arg > 0)
   Arg                 is an integer, which is 0 if the list is a list of
	    dvarints or greater 0 if the list consists of terms of
	    arity greater than Arg, the value Arg indicates the
	    selected argument of the term
   Select              is a predefined selection method or the name of a
	    predicate of arity 2.  Predefined methods are input_order,
	    first_fail, smallest, largest, occurrence,
	    most_constrained, max_regret, anti_first_fail
   Choice              is the name of a predicate of arity 1 or a term
	    with two arguments with the same functor as a predicate of
	    arity 3.  Some names are already predefined as special
	    cases and are handled without a meta-call:  indomain,
	    indomain_min, indomain_max, indomain_middle,
	    indomain_median, indomain_split, indomain_random,
	    indomain_interval
   Method              is one of the following:  complete,
	    bbs(Steps:integer), lds(Disc:integer),
	    credit(Credit:integer, Extra:integer or bbs(Steps:integer)
	    or lds(Disc:integer)), dbs(Level:integer, Extra:integer or
	    bbs(Steps:integer) or lds(Disc:integer))
   Option              is a list of option terms.  Currently recognized
	   are backtrack(-N), node(++Call), nodes(++N)

Type
   library(fd_search)

Description
Search/6 provides a generic interface to a set of different search
methods.  It can currently be used with either the finite domains (if
loaded via lib(fd_search)), or integer IC domains (if loaded via
lib(ic_search)).  By changing the Method argument, different partial
search algorithms (and their parameters) can be selected and
controlled. The search predicate also provides a number of pre-defined
variable selection methods (to choose which variable will be assigned next)
and some pre-defined value assignment methods (to try out the possible
values for the selected variable in some heuristic order), but user-defined
methods can be used in their place as well.  In order to allow more
structure in the application program, it is possible to pass a list of
terms rather than only a list of domain variables. In this way all
information about some entity can be easily grouped together. It also
allows more complex labeling methods which combine the assignment of
multiple variables (like a preference value and a decision variable).


All search methods use a stable selection method. If several entries have the 
same heuristic value, then the first one is selected.
The rest of the list is equal to the original list with the selected entry removed,
the order of the non-selected entries does not change.

Integer values are not treated differently from the domain variables,
they are selected only if their heuristic value is better than those of the 
other entries.

The pre-defined selection methods use the following criteria:

input_order the first entry in the list is selected
first_fail the entry with the smallest domain size is selected
anti_first_fail the entry with the largest domain size is selected
smallest the entry with the smallest value in the domain is selected
largest the entry with the largest value in the domain is selected
occurrence the entry with the largest number of attached constraints is selected
most_constrained the entry with the smallest domain size is selected. If several entries have the same domain size, the entry with the largest number of attached constraints is selected.
max_regret the entry with the largest difference between the smallest and second smallest value in the domain is selected. This method is typically used if the variable represents a cost, and we are interested in the choice which could increase overall cost the most if the best possibility is not taken. Unfortunately, the implementation sometimes does not always work. If two decision variables incur the same minimal cost, the regret is not calculated as zero, but as the difference from this minimal value to the next greater value.

Any other name is taken as the name of a user-defined predicate of arity 2
which is expected to compute a selection criterion (typically a number), e.g.

my_select(X,Criterion) :-
	...	% compute Criterion from variable X

The variable-selection will then select the variable with the lowest
value of Criterion.  If several variables have the same value, the first
one is selected.

The pre-defined choice methods have the following meaning:

indomain
uses the built-in indomain/1.  Values are tried in increasing order. 
On failure, the previously tested value is not removed.

indomain_min
Values are tried in increasing order.  On failure, the previously
tested value is removed.  The values are tested in the same order as
for indomain, but backtracking may occur earlier.

indomain_max
Values are tried in decreasing order.  On failure, the previously
tested value is removed.

indomain_middle
Values are tried beginning from the middle of the domain.  On failure,
the previously tested value is removed.

indomain_median
Values are tried beginning from the median value of the domain.  On
failure, the previously tested value is removed.

indomain_split
Values are tried by succesive domain splitting.  On failure, the
previously tried interval is removed.  This enumerates values in the
same order as indomain or indomain_min, but may fail earlier.

indomain_random
Values are tried in a random order.  On backtracking, the previously
tried value is removed.  Using this rutine may lead to unreproducable
results, as another call wil create random numbers in a different
sequence.  This method uses the built-in random/1 to create
random numbers, seed/1 can be used to force the same number
generation sequence in another run.

indomain_interval
If the domain consists of several intervals, we first branch on the
choice of the interval.  For one interval, we use domain
splitting.


Any other name is taken as the name of a user-defined predicate of
arity 1, e.g.

my_choice(X) :-
	...	% make a choice on variable X

Alternatively, a term with 2 arguments can be given as the choice-method,
e.g. my_choice(FirstIn,LastOut). this will lead to the invocation of a
choice predicate with arity 3, e.g.

my_choice(X,In,Out) :-
	...	% make a choice on variable X, using In-Out

This allows user-defined state to be transferred between the subsequent
invocations of the choice-predicate.

The different search methods are

complete
a complete search routine which explores all alternative choices.

bbs(Steps)
The bounded backtracking search allows Steps
backtracking steps.

lds(Disc)
A form of the  limited discrepancy search .  This method
iteratively tries 0, 1, 2 ..  Disc changes against the
heuristic (first) value.  Typical values are between 1 and 3 (which
already may create too many alternatives for large problems).  The
original LDS paper stated that the discrepancy to be tested first
should be at the top of the tree.  Our implementation tries the first
discrepancy at the bottom of the tree.  This means that solutions may
be found in a different order compared to the original algorithm. 
This change is imposed by the evaluation strategy used and can not be
easily modified.

credit(Credit, bbs(Steps))
The credit based search explores the top of the search tree
completely.  Initially, a given number of credits (Credit) are
given.  At each choice point, the first alternative gets half of the
available credit, the second alternative half of the remaining credit,
and so on.  When the credit run out, the system switches to another
search routine, here bbs.  In each of these bounded backtracking
searches Steps backtracking steps are allowed before returning
to the top most part of the tree and choosing the next remaining
candidate.  A good value for Steps is 5, a value of 0 forces a
deterministic search using the heuristic.  Typical values for
Credit are either N or N*N, where N is the number of entries in
the list.

credit(Credit, lds(Disc))
like the one above, but using lds when the credit runs out. 
Typically, only one (perhaps 2) discrepancies should be allowed.

dbs(Level, bbs(Steps))
The depth bounded search explores the first Level
choices in the search tree completely, i.e.  it tries all values for
the first Level selected variables.  After that, it switches to
another search method, here bbs.  In each of these searches,
Steps backtracking steps are allowed.  

dbs(Level, lds(Disc)
like the method above, but switches to lds after the first
Level variables.

The option list is used to pass additional parameters to and from the
procedure.  The currently recognized options are:

backtrack(-N)
returns the number of backtracking steps used in the search
routine

nodes(++N)
sets an upper limit on the number of nodes explored in the search.  If
the given limit is exceeded, the search routine stops the exploration
of the search tree.

node(daVinci)
create a drawing of the search tree using the daVinci grapg drawing
tool.  Each node of the search tree is shown as a node in the tree. 
The label of the node is the selected term of the list.

node(daVinci(++Call))
as the previous option, it creates a drawing of the search tree using
the daVinci graph drawing tool.  But instead of using the complete
selected term as the label, it call the predicate Call/2 to
choose which part of the selected term to display.



Resatisfiable
   yes

Fail Conditions
   Fails if the search tree generated does not contain any solution. 
For partial search methods, this does not mean that the problem does not 
have a solution, but only that the part of the tree generated did not 
contain one.

Examples
   
top:-
	length(L,8),
	L :: 1..8,
	search(L,0,input_order,indomain,complete,[]).

top:-
	length(L,8),
	L :: 1..8,
	search(L,0,input_order,indomain,bbs(15),[]).

top:-
	length(L,8),
	L :: 1..8,
	search(L,0,input_order,indomain,lds(2),[]).

top:-
	length(L,8),
	L :: 1..8,
	search(L,0,input_order,indomain,credit(64,bbs(5)),[]).

top:-
	length(L,8),
	L :: 1..8,
	search(L,0,input_order,indomain,dbs(2,lds(1)),[]).

% a more complex example with different methods and heuristics
% the list to be assigned is a list of terms queen/2

:- local struct(queen(place,var)).

top:-
	member(Method,[complete,lds(2),credit(64,5),bbs(1000),dbs(5,10)]),
	member(Select,[first_fail,most_constrained,input_order]),
	member(Choice,[indomain,
	               indomain_min,
		       indomain_max,
		       indomain_middle,
		       indomain_median,
		       indomain_split,
		       indomain_random]),
	writeln(queen(Method,Select,Choice)),
	once(queen_credit(64,Select,Choice,Method,L,Back)),
	writeln(L),
	writeln(backtrack(Back)),
	fail.
top:-
	nl.

queen_credit(N,Select,Choice,Method,L,Back):-
	create_queens(1,N,Queens,L),
	setup(L),
	rearrange(Queens,Queens,[],[],Queens1),
	search(Queens1, var of queen, Select, Choice, Method, [backtrack(Back)]).

rearrange([],Last,Last,Res,Res).
rearrange([_],[Mid|Last],Last,Res,[Mid|Res]).
rearrange([_,_|S],[H|T],A1,In,Res):-
	rearrange(S,T,[A|A1],[H,A|In],Res).

create_queens(N,M,[],[]):-
	N > M,
	!.
create_queens(N,M,[queen with [place:N,var:X]|T],[X|L]):-
	X :: 1..M,
	N1 is N+1,
	create_queens(N1,M,T,L).

setup([]).
setup([H|T]):-
	setup1(H,T,1),
	setup(T).

setup1(_,[],_).
setup1(X,[Y|R],N):-
	X #\= Y,
	X #\= Y + N,
	Y #\= X + N,
	N1 is N+1,
	setup1(X,R,N1).


% this example shows how to pass information from one assignment step 
% to the next
% this uses a term of two arguments as the choice argument
% The example also shows the use of the option argument:
% the search tree generated is drawn with the daVinci graph drawing tool
% and the search is limited to 1000 nodes.
% The number of backtracking steps is returned in the variables Back.
:-local struct(country(name,color)).

top:-
	countries(C),
	create_countries(C,Countries,Vars),
	findall(n(A,B),n(A,B),L),
	setup(L,Countries),
	search(Countries,
	       color of country, % select based on this variable
	       most_constrained,
	       assign([1,2,3,4],Out), % this calls assign/3
	       complete,
	       [backtrack(Back),node(daVinci),nodes(1000)]),
	writeln(Vars),
	writeln(Back),
	writeln(Out).

create_countries([],[],[]).
create_countries([C|C1],[country with [name:C, color:V]|R1],[V|V1]):-
	V :: 1..4,
	create_countries(C1,R1,V1).

setup([],_L).
setup([n(A,B)|N1],L):-
	member(country with [name:A, color:Av],L),
	member(country with [name:B, color:Bv],L),
	Av #\= Bv,
	setup(N1,L).

% this is the choice predicate
% the first argument is the complete selected term
% the second is the input argument
% the third is the output argument
% here we pass a list of values and rotate this list from one step to the next
assign(country with color:X,L,L1):-
	rotate(L,L1),
	member(X,L).

rotate([A,B,C,D],[B,C,D,A]).

% another example of argument passing 
% here each entry gets the same information
% it is passed unchanged from one level to the next

top:-
	...
	length(L,N),
	L :: 1..10,
	...
        search(L,
	       0,
	       most_constrained,
	       % pass two lists as the In argument of assign
	       % try the odd numbers before the even numbers
	       assign([1,3,5,7,9]-[2,4,6,8,10],_), 
	       complete,[]),
	...

% this is the assignment routine
% the first argument is a 
% Pass the In argument as the Out argument
% try values from list L1 before values from list L2
assign(X,L1-L2,L1-L2):-
	member(X,L1);member(X,L2).

% and another example from square placement
% alternatively try minimal and maximal values first

:-local struct(square(x,y,size)).

top:-
	data(L),
	create_squares(L,Squares),
	...
        search(Squares,
	       0, % this value does not matter if input_order is chosen
	       input_order,
	       assign(min,_),
	       complete,
	       []),
	...

% the assignment routine
% alternate between min and max for consecutive levels in the search
assign(square with [x:X,y:Y],Type,Type1):-
	swap(Type,Type1),
	indomain(X,Type),
	indomain(Y,Type).

swap(max,min).
swap(min,max).

% this example shows that the choice routine may consist of several clauses
% the idea comes from a graph coloring heuristic

top:-
	length(L,N),
	L :: 1..100,
	...
        search(L,
	       0,
	       most_constrained,
	       assign(0,K), The In argument is the highest color used so far
	       complete,[]),
	...


% assign variable X either to one of the colors 1..K 
% which have already been used, or to the new color K+1
% we do not need to try other values K+2 etc, as this is a symmetry that
% we can avoid
assign(X,K,K):-
	X #=< K,
	indomain(X).
assign(K1,K,K1):-
	K1 is K+1.



See Also
   fd : indomain / 1, fdplex : indomain / 1, ic : indomain / 1, indomain / 2, fd : labeling / 1, fdplex : labeling / 1, ic : labeling / 1, fd : deleteff / 3, fd : deleteffc / 3
