mirror of
https://github.com/Mercury-Language/mercury.git
synced 2026-04-16 01:43:35 +00:00
extras/curs/curs.m:
extras/curs/curs.panel.m:
Make panel a separate submodule of curs, not a nested submodule.
extras/base64/base64.m:
extras/curses/mcurses.basics.m:
extras/curses/mcurses.m:
extras/curses/mcurses.misc.m:
extras/curses/mcurses.user.m:
extras/gator/evolve.m:
extras/gator/genotype.m:
extras/gator/phenotype.m:
extras/gator/tausworthe3.m:
extras/monte/dots.m:
extras/monte/geom.m:
extras/monte/hg.m:
extras/monte/monte.m:
extras/monte/rnd.m:
extras/moose/grammar.m:
extras/moose/moose.m:
extras/mopenssl/mopenssl.m:
extras/net/echo.m:
extras/net/errno.m:
extras/net/getaddrinfo.m:
extras/net/net.m:
extras/net/netdb.m:
extras/net/sockets.m:
extras/net/streams.m:
extras/net/tcp.m:
extras/net/test_lookups.m:
extras/net/types.m:
extras/odbc/odbc.m:
extras/odbc/odbc_test.m:
extras/references/README:
extras/references/reference.m:
extras/references/scoped_update.m:
extras/solver_types/library/any.m:
extras/solver_types/library/any_array.m:
extras/solver_types/library/any_assoc_list.m:
extras/solver_types/library/any_list.m:
extras/solver_types/library/any_map.m:
extras/solver_types/library/any_tree234.m:
extras/solver_types/library/any_util.m:
extras/trail/trail.m:
extras/trailed_update/samples/interpreter.m:
extras/trailed_update/samples/vqueens.m:
extras/trailed_update/tests/var_test.m:
extras/trailed_update/tr_array.m:
extras/trailed_update/tr_store.m:
extras/trailed_update/trailed_update.m:
extras/trailed_update/unsafe.m:
extras/trailed_update/var.m:
Bring programming style up to date.
163 lines
5.5 KiB
Mathematica
163 lines
5.5 KiB
Mathematica
%---------------------------------------------------------------------------%
|
|
% vim: ft=mercury ts=4 sw=4 et
|
|
%---------------------------------------------------------------------------%
|
|
% Copyright (C) 1998-2000,2002-2004, 2006 The University of Melbourne.
|
|
% Copyright (C) 2018, 2022 The Mercury team.
|
|
% This file is distributed under the terms specified in COPYING.LIB.
|
|
%---------------------------------------------------------------------------%
|
|
%
|
|
% File: reference.m.
|
|
% Authors: pets (Peter Schachte).
|
|
% Stability: low.
|
|
% Purpose: A backtrackably modifiable storage type.
|
|
%
|
|
% This module defines a type which is, in essence, a ``box'' into which you can
|
|
% put any term, and later destructively replace the contents with something
|
|
% else. The store module in the standard library provides a clean way to do
|
|
% exactly the same thing; the difference is that the reference module allows
|
|
% you to do it without threading a store through your code. The price for this
|
|
% convenience is that you must mark all predicates that create or modify a
|
|
% reference, and all the predicates that call them, and so on, as impure. This
|
|
% is probably more inconvenient than just threading the store through your
|
|
% code, so this module should probably only be used in exceptional
|
|
% circumstances.
|
|
%
|
|
% This module is implemented using the trailing features described in the
|
|
% "Trailing" section of the "Implementation-dependent extensions" chapter of
|
|
% the Mercury Language Reference Manual. This means that in order to use this
|
|
% module, you *must* compile with the --use-trail switch. The easiest way to
|
|
% do this is to include the line
|
|
%
|
|
% GRADEFLAGS=--use-trail
|
|
%
|
|
% in your Mmakefile.
|
|
%
|
|
%---------------------------------------------------------------------------%
|
|
|
|
:- module reference.
|
|
:- interface.
|
|
|
|
% A backtrackably destructively modifiable reference type.
|
|
%
|
|
:- type reference(T).
|
|
|
|
% Create a new reference given a term for it to (initially) refer to.
|
|
%
|
|
:- impure pred new_reference(T::in, reference(T)::out) is det.
|
|
|
|
% Get the value currently referred to by a reference.
|
|
%
|
|
:- semipure pred value(reference(T)::in, T::out) is det.
|
|
|
|
% (Backtrackably) modify a reference to refer to a new object.
|
|
%
|
|
:- impure pred update(reference(T)::in, T::in) is det.
|
|
|
|
% init(Ref, Value):
|
|
% Initialise a reference Ref to have value Value.
|
|
% This is for use with user-declared ME_References (see c_reference.h), and
|
|
% must be called before using such a reference. Attempting to access the
|
|
% reference before it is initialised or after the init call is backtracked
|
|
% is undefined.
|
|
%
|
|
:- impure pred init(reference(T)::in, T::in) is det.
|
|
|
|
% from_c_pointer(CPointer) = Ref:
|
|
% Convert a c_pointer to a reference.
|
|
%
|
|
:- func from_c_pointer(c_pointer) = reference(T).
|
|
|
|
% to_c_pointer(Ref) = CPointer
|
|
% Convert a reference to a c_pointer.
|
|
%
|
|
:- func to_c_pointer(reference(T)) = c_pointer.
|
|
|
|
%---------------------------------------------------------------------------%
|
|
%---------------------------------------------------------------------------%
|
|
|
|
:- implementation.
|
|
|
|
% This type is implemented in C.
|
|
:- type reference(T)
|
|
---> reference(c_pointer).
|
|
|
|
:- pragma foreign_decl("C", "#include ""c_reference.h""").
|
|
|
|
:- pragma inline(pred(new_reference/2)).
|
|
new_reference(X, reference(Ref)) :-
|
|
impure new_reference_2(X, Ref).
|
|
|
|
:- impure pred new_reference_2(T::in, c_pointer::out) is det.
|
|
:- pragma inline(new_reference_2/2).
|
|
:- pragma foreign_proc("C",
|
|
new_reference_2(X::in, Ref::out),
|
|
[will_not_call_mercury],
|
|
"
|
|
MR_incr_hp(Ref, (sizeof(ME_Reference) + sizeof(MR_Word) - 1) /
|
|
sizeof(MR_Word));
|
|
((ME_Reference *) Ref)->value = (void *) X;
|
|
// Use MR_null_choicepoint_id here instead of
|
|
// MR_current_choicepoint_id, in case this is called from
|
|
// a tabled pred/func -- even if it isn't, this will only
|
|
// result in one additional (redundant) entry on the trail.
|
|
((ME_Reference *) Ref)->id = MR_null_choicepoint_id();
|
|
").
|
|
|
|
:- pragma inline(value/2).
|
|
value(reference(Ref), X) :-
|
|
semipure value_2(Ref, X).
|
|
|
|
:- semipure pred value_2(c_pointer::in, T::out) is det.
|
|
:- pragma inline(value_2/2).
|
|
:- pragma foreign_proc("C",
|
|
value_2(Ref::in, X::out),
|
|
[promise_semipure, will_not_call_mercury],
|
|
"
|
|
X = (MR_Word) ((ME_Reference *) Ref)->value;
|
|
").
|
|
|
|
:- pragma inline(update/2).
|
|
update(reference(Ref), X) :-
|
|
impure update_2(Ref, X).
|
|
|
|
:- impure pred update_2(c_pointer::in, T::in) is det.
|
|
:- pragma inline(pred(update_2/2)).
|
|
:- pragma foreign_proc("C",
|
|
update_2(Ref::in, X::in),
|
|
[will_not_call_mercury],
|
|
"
|
|
ME_Reference *ref = (ME_Reference *) Ref;
|
|
if (ref->id != MR_current_choicepoint_id()) {
|
|
MR_trail_current_value((MR_Word *) (&ref->value));
|
|
MR_trail_current_value((MR_Word *) (&ref->id));
|
|
ref->id = MR_current_choicepoint_id();
|
|
}
|
|
ref->value = (void *) X;
|
|
").
|
|
|
|
:- pragma inline(pred(init/2)).
|
|
init(reference(Ref), X) :-
|
|
impure init_2(Ref, X).
|
|
|
|
:- impure pred init_2(c_pointer::in, T::in) is det.
|
|
:- pragma inline(init_2/2).
|
|
:- pragma foreign_proc("C",
|
|
init_2(Ref::in, X::in),
|
|
[will_not_call_mercury],
|
|
"
|
|
((ME_Reference *) Ref)->value = (void *) X;
|
|
((ME_Reference *) Ref)->id = MR_current_choicepoint_id();
|
|
").
|
|
|
|
:- pragma inline(func(from_c_pointer/1)).
|
|
|
|
from_c_pointer(CPointer) = reference(CPointer).
|
|
|
|
:- pragma inline(func(to_c_pointer/1)).
|
|
|
|
to_c_pointer(reference(CPointer)) = CPointer.
|
|
|
|
%---------------------------------------------------------------------------%
|
|
:- end_module reference.
|
|
%---------------------------------------------------------------------------%
|