Files
mercury/extras/trailed_update/tr_array.m
Fergus Henderson aa985ccb0a Various bug fixes and enhancements to var.m and other stuff in
Estimated hours taken: 6

Various bug fixes and enhancements to var.m and other stuff in
extras/trailed_update.

extras/trailed_update/var.m:
	Add code to detect floundering.
	Change the code for debug_freeze/3 to avoid a floundering warning.
	Add new predicates debug_freeze/4 (to go with debug_freeze/3)
	and dump_var/3.
	Fix a bug in freeze/3 (higher-order terms are monomorphic, so
	there is no need to explicitly pass type-infos when calling them).
	Fix a bug in wakeup_delayed_goals/2 (we need to set the binding
	to ground _before_ waking up the goals).

extras/trailed_update/Mmakefile:
	Add `ML = ml --mercury-libs shared', so that it will work on Linux.

extras/trailed_update/samples/Mmakefile:
	Make it work with shared libraries on Linux.

extras/trailed_update/samples/tests.m:
extras/trailed_update/samples/tests.exp:
	Some test cases for var.m.

extras/trailed_update/tr_store.m:
	Add a prototype for ML_compare_typeinfo().

extras/trailed_update/unsafe.m:
	Change the C code for unsafe_perform_io to match the new
	interface for exported Mercury procedures that do I/O --
	there is no longer any need to pass dummy io__state arguments
	in the C code.
1997-10-06 18:46:58 +00:00

499 lines
16 KiB
Mathematica

%-----------------------------------------------------------------------------%
% Copyright (C) 1993-1995, 1997 The University of Melbourne.
% This file may only be copied under the terms of the GNU Library General
% Public License - see the file COPYING.LIB in the Mercury distribution.
%-----------------------------------------------------------------------------%
% File: tr_array.m
% Main authors: fjh
% Stability: medium
% This module provides backtrackable destructive update operations on arrays.
%-----------------------------------------------------------------------------%
%-----------------------------------------------------------------------------%
:- module tr_array.
:- interface.
:- import_module array, list, std_util.
%-----------------------------------------------------------------------------%
%
% Operations that perform backtrackable destructive update.
%
% tr_array__set sets the nth element of an array, and returns the
% resulting array (good opportunity for destructive update ;-).
% It is an error if the index is out of bounds.
:- pred tr_array__set(array(T), int, T, array(T)).
:- mode tr_array__set(array_mdi, in, in, array_muo) is det.
% tr_array__semidet_set sets the nth element of an array,
% and returns the resulting array.
% It fails if the index is out of bounds.
:- pred tr_array__semidet_set(array(T), int, T, array(T)).
:- mode tr_array__semidet_set(array_mdi, in, in, array_muo) is semidet.
%-----------------------------------------------------------------------------%
%
% "mui" (Mostly Unique Input) versions of standard array operations.
% It is necessary to use these ones rather than the standard array
% operations when dealing with mostly-unique values, to preserve
% "mostly_unique"-ness.
%
% tr_array__min returns the lower bound of the array.
% Note: in this implementation, the lower bound is always zero.
:- pred tr_array__min(array(_T), int).
:- mode tr_array__min(array_mui, out) is det.
:- mode tr_array__min(in, out) is det.
% tr_array__max returns the upper bound of the array.
:- pred tr_array__max(array(_T), int).
:- mode tr_array__max(array_mui, out) is det.
:- mode tr_array__max(in, out) is det.
% tr_array__size returns the length of the array,
% i.e. upper bound - lower bound + 1.
:- pred tr_array__size(array(_T), int).
:- mode tr_array__size(array_mui, out) is det.
:- mode tr_array__size(in, out) is det.
% tr_array__bounds returns the upper and lower bounds of an array.
% Note: in this implementation, the lower bound is always zero.
:- pred tr_array__bounds(array(_T), int, int).
:- mode tr_array__bounds(array_mui, out, out) is det.
:- mode tr_array__bounds(in, out, out) is det.
% tr_array__in_bounds checks whether an index is in the bounds
% of an array.
:- pred tr_array__in_bounds(array(_T), int).
:- mode tr_array__in_bounds(array_mui, in) is semidet.
:- mode tr_array__in_bounds(in, in) is semidet.
%-----------------------------------------------------------------------------%
% tr_array__lookup returns the Nth element of an array.
% It is an error if the index is out of bounds.
:- pred tr_array__lookup(array(T), int, T).
:- mode tr_array__lookup(array_mui, in, out) is det.
:- mode tr_array__lookup(in, in, out) is det.
% tr_array__semidet_lookup returns the Nth element of an array.
% It fails if the index is out of bounds.
:- pred tr_array__semidet_lookup(array(T), int, T).
:- mode tr_array__semidet_lookup(array_mui, in, out) is semidet.
:- mode tr_array__semidet_lookup(in, in, out) is semidet.
% tr_array__slow_set sets the nth element of an array,
% and returns the resulting array. The initial array is not
% required to be unique, so the implementation may not be able to use
% destructive update.
% It is an error if the index is out of bounds.
:- pred tr_array__slow_set(array(T), int, T, array(T)).
:- mode tr_array__slow_set(array_mui, in, in, array_uo) is det.
:- mode tr_array__slow_set(in, in, in, array_uo) is det.
% tr_array__semidet_slow_set sets the nth element of an array,
% and returns the resulting array. The initial array is not
% required to be unique, so the implementation may not be able to use
% destructive update.
% It fails if the index is out of bounds.
:- pred tr_array__semidet_slow_set(array(T), int, T, array(T)).
:- mode tr_array__semidet_slow_set(array_mui, in, in, array_uo) is semidet.
:- mode tr_array__semidet_slow_set(in, in, in, array_uo) is semidet.
% tr_array__copy(Array0, Array):
% Makes a new unique copy of an array.
:- pred tr_array__copy(array(T), array(T)).
:- mode tr_array__copy(array_mui, array_uo) is det.
:- mode tr_array__copy(in, array_uo) is det.
% tr_array__resize(Array0, Size, Init, Array):
% The array is expanded or shrunk to make it fit
% the new size `Size'. Any new entries are filled
% with `Init'.
:- pred tr_array__resize(array(T), int, T, array(T)).
:- mode tr_array__resize(array_mui, in, in, array_uo) is det.
:- mode tr_array__resize(in, in, in, array_uo) is det.
% tr_array__shrink(Array0, Size, Array):
% The array is shrunk to make it fit the new size `Size'.
% It is an error if `Size' is larger than the size of `Array0'.
:- pred tr_array__shrink(array(T), int, array(T)).
:- mode tr_array__shrink(array_mui, in, array_uo) is det.
:- mode tr_array__shrink(in, in, array_uo) is det.
% tr_array__to_list takes an array and returns a list containing
% the elements of the array in the same order that they
% occurred in the array.
:- pred tr_array__to_list(array(T), list(T)).
:- mode tr_array__to_list(array_mui, out) is det.
:- mode tr_array__to_list(in, out) is det.
% tr_array__fetch_items takes an array and a lower and upper
% index, and places those items in the array between these
% indices into a list. It is an error if either index is
% out of bounds.
:- pred tr_array__fetch_items(array(T), int, int, list(T)).
:- mode tr_array__fetch_items(array_mui, in, in, out) is det.
:- mode tr_array__fetch_items(in, in, in, out) is det.
% tr_array__bsearch takes an array, an element to be found
% and a comparison predicate and returns the position of
% the element in the array. Assumes the array is in sorted
% order. Fails if the element is not present. If the
% element to be found appears multiple times, the index of
% the first occurrence is returned.
:- pred tr_array__bsearch(array(T), T, pred(T, T, comparison_result),
maybe(int)).
:- mode tr_array__bsearch(array_mui, in, pred(in, in, out) is det, out)
is det.
:- mode tr_array__bsearch(in, in, pred(in, in, out) is det, out) is det.
%-----------------------------------------------------------------------------%
%-----------------------------------------------------------------------------%
:- implementation.
:- import_module int.
/****
lower bounds other than zero are not supported
% tr_array__resize takes an array and new lower and upper bounds.
% the array is expanded or shrunk at each end to make it fit
% the new bounds.
:- pred tr_array__resize(array(T), int, int, array(T)).
:- mode tr_array__resize(in, in, in, out) is det.
****/
%-----------------------------------------------------------------------------%
% Arrays are implemented in array.m using the C interface.
% The C type which defines the representation of arrays is
% MR_ArrayType; it is defined in runtime/type_info.h.
%-----------------------------------------------------------------------------%
:- pragma c_code(tr_array__set(Array0::array_mdi, Index::in, Item::in,
Array::array_muo),
will_not_call_mercury,
"{
MR_ArrayType *array = (MR_ArrayType *) Array0;
if ((Unsigned) Index >= (Unsigned) array->size) {
fatal_error(""tr_array__set: array index out of bounds"");
}
MR_trail_current_value(&array->elements[Index]);
array->elements[Index] = Item; /* destructive update! */
Array = Array0;
}").
tr_array__semidet_set(Array0, Index, Item, Array) :-
tr_array__in_bounds(Array0, Index),
tr_array__set(Array0, Index, Item, Array).
%-----------------------------------------------------------------------------%
%-----------------------------------------------------------------------------%
:- pragma c_code(tr_array__min(Array::array_mui, Min::out),
will_not_call_mercury,
"
/* Array not used */
Min = 0;
").
:- pragma c_code(tr_array__min(Array::in, Min::out),
will_not_call_mercury,
"
/* Array not used */
Min = 0;
").
:- pragma c_code(tr_array__max(Array::array_mui, Max::out),
will_not_call_mercury,
"
Max = ((MR_ArrayType *)Array)->size - 1;
").
:- pragma c_code(tr_array__max(Array::in, Max::out),
will_not_call_mercury,
"
Max = ((MR_ArrayType *)Array)->size - 1;
").
tr_array__bounds(Array, Min, Max) :-
tr_array__min(Array, Min),
tr_array__max(Array, Max).
%-----------------------------------------------------------------------------%
:- pragma c_code(tr_array__size(Array::array_mui, Max::out),
will_not_call_mercury,
"
Max = ((MR_ArrayType *)Array)->size;
").
:- pragma c_code(tr_array__size(Array::in, Max::out),
will_not_call_mercury,
"
Max = ((MR_ArrayType *)Array)->size;
").
%-----------------------------------------------------------------------------%
tr_array__in_bounds(Array, Index) :-
tr_array__bounds(Array, Min, Max),
Min =< Index, Index =< Max.
tr_array__semidet_lookup(Array, Index, Item) :-
tr_array__in_bounds(Array, Index),
tr_array__lookup(Array, Index, Item).
tr_array__semidet_slow_set(Array0, Index, Item, Array) :-
tr_array__in_bounds(Array0, Index),
tr_array__slow_set(Array0, Index, Item, Array).
tr_array__slow_set(Array0, Index, Item, Array) :-
tr_array__copy(Array0, Array1),
array__set(Array1, Index, Item, Array).
%-----------------------------------------------------------------------------%
:- pragma c_code(tr_array__lookup(Array::array_mui, Index::in, Item::out),
will_not_call_mercury,
"{
MR_ArrayType *array = (MR_ArrayType *) Array;
if ((Unsigned) Index >= (Unsigned) array->size) {
fatal_error(""tr_array__lookup: ""
""array index out of bounds"");
}
Item = array->elements[Index];
}").
:- pragma c_code(tr_array__lookup(Array::in, Index::in, Item::out),
will_not_call_mercury,
"{
MR_ArrayType *array = (MR_ArrayType *) Array;
if ((Unsigned) Index >= (Unsigned) array->size) {
fatal_error(""tr_array__lookup: array index out of bounds"");
}
Item = array->elements[Index];
}").
%-----------------------------------------------------------------------------%
:- pragma c_header_code("
MR_ArrayType * ML_tr_resize_array(MR_ArrayType *old_array,
Integer array_size, Word item);
").
:- pragma c_code("
MR_ArrayType *
ML_tr_resize_array(MR_ArrayType *old_array, Integer array_size,
Word item)
{
Integer i;
MR_ArrayType* array;
Integer elements_to_copy;
elements_to_copy = old_array->size;
if (elements_to_copy == array_size) return old_array;
if (elements_to_copy > array_size) {
elements_to_copy = array_size;
}
array = (MR_ArrayType *) make_many(Word, array_size + 1);
array->size = array_size;
for (i = 0; i < elements_to_copy; i++) {
array->elements[i] = old_array->elements[i];
}
for (; i < array_size; i++) {
array->elements[i] = item;
}
/*
** since the mode on the old array is `array_mdi', it is NOT safe to
** deallocate the storage for it
*/
return array;
}
").
:- pragma c_code(
tr_array__resize(Array0::array_mui, Size::in, Item::in,
Array::array_uo),
will_not_call_mercury,
"
Array = (Word) ML_tr_resize_array((MR_ArrayType *) Array0, Size, Item);
").
:- pragma c_code(
tr_array__resize(Array0::in, Size::in, Item::in,
Array::array_uo),
will_not_call_mercury,
"
Array = (Word) ML_tr_resize_array((MR_ArrayType *) Array0, Size, Item);
").
%-----------------------------------------------------------------------------%
:- pragma c_header_code("
MR_ArrayType * ML_tr_shrink_array(MR_ArrayType *old_array,
Integer array_size);
").
:- pragma c_code("
MR_ArrayType *
ML_tr_shrink_array(MR_ArrayType *old_array, Integer array_size)
{
Integer i;
MR_ArrayType* array;
Integer old_array_size;
old_array_size = old_array->size;
if (old_array_size == array_size) return old_array;
if (old_array_size < array_size) {
fatal_error(""tr_array__shrink: can't shrink to a larger size"");
}
array = (MR_ArrayType *) make_many(Word, array_size + 1);
array->size = array_size;
for (i = 0; i < array_size; i++) {
array->elements[i] = old_array->elements[i];
}
/*
** since the mode on the old array is `array_mdi', it is NOT safe to
** deallocate the storage for it
*/
return array;
}
").
:- pragma c_code(
tr_array__shrink(Array0::array_mui, Size::in, Array::array_uo),
will_not_call_mercury,
"
Array = (Word) ML_tr_shrink_array(
(MR_ArrayType *) Array0, Size);
").
:- pragma c_code(
tr_array__shrink(Array0::in, Size::in, Array::array_uo),
will_not_call_mercury,
"
Array = (Word) ML_tr_shrink_array(
(MR_ArrayType *) Array0, Size);
").
%-----------------------------------------------------------------------------%
:- pragma c_header_code("
MR_ArrayType *ML_tr_copy_array(MR_ArrayType *old_array);
").
:- pragma c_code("
MR_ArrayType *
ML_tr_copy_array(MR_ArrayType *old_array)
{
/*
** Any changes to this function will probably also require
** changes to deepcopy() in runtime/deep_copy.c.
*/
Integer i;
MR_ArrayType* array;
Integer array_size;
array_size = old_array->size;
array = MR_make_array(array_size);
array->size = array_size;
for (i = 0; i < array_size; i++) {
array->elements[i] = old_array->elements[i];
}
return array;
}
").
:- pragma c_code(
tr_array__copy(Array0::array_mui, Array::array_uo),
will_not_call_mercury,
"
Array = (Word) ML_tr_copy_array((MR_ArrayType *) Array0);
").
:- pragma c_code(
tr_array__copy(Array0::in, Array::array_uo),
will_not_call_mercury,
"
Array = (Word) ML_tr_copy_array((MR_ArrayType *) Array0);
").
%-----------------------------------------------------------------------------%
tr_array__to_list(Array, List) :-
tr_array__bounds(Array, Low, High),
tr_array__fetch_items(Array, Low, High, List).
%-----------------------------------------------------------------------------%
tr_array__fetch_items(Array, Low, High, List) :-
(
Low > High
->
List = []
;
Low1 is Low + 1,
tr_array__fetch_items(Array, Low1, High, List0),
tr_array__lookup(Array, Low, Item),
List = [Item|List0]
).
%-----------------------------------------------------------------------------%
tr_array__bsearch(A, El, Compare, Result) :-
tr_array__bounds(A, Lo, Hi),
tr_array__bsearch_2(A, Lo, Hi, El, Compare, Result).
:- pred tr_array__bsearch_2(array(T), int, int, T,
pred(T, T, comparison_result), maybe(int)).
:- mode tr_array__bsearch_2(in, in, in, in, pred(in, in, out) is det,
out) is det.
tr_array__bsearch_2(Array, Lo, Hi, El, Compare, Result) :-
Width is Hi - Lo,
% If Width < 0, there is no range left.
( Width < 0 ->
Result = no
;
% If Width == 0, we may just have found our element.
% Do a Compare to check.
( Width = 0 ->
tr_array__lookup(Array, Lo, X),
( call(Compare, El, X, (=)) ->
Result = yes(Lo)
;
Result = no
)
;
% Otherwise find the middle element of the range
% and check against that.
Mid is (Lo + Hi) >> 1, % `>> 1' is hand-optimized `div 2'.
tr_array__lookup(Array, Mid, XMid),
call(Compare, XMid, El, Comp),
( Comp = (<),
Mid1 is Mid + 1,
tr_array__bsearch_2(Array, Mid1, Hi, El, Compare, Result)
; Comp = (=),
tr_array__bsearch_2(Array, Lo, Mid, El, Compare, Result)
; Comp = (>),
Mid1 is Mid - 1,
tr_array__bsearch_2(Array, Lo, Mid1, El, Compare, Result)
)
)
).
%-----------------------------------------------------------------------------%
%-----------------------------------------------------------------------------%