Files
mercury/library/term_io.m
Julien Fischer e3cc72cb0b Fix bug in term_io.format_term_nl/5.
library/term_io.m:
     foramt_term_nl/5 should call format_term_nl_with_op_table/6, *not*
     format_term_with_op_table/6, otherwise the terminating ".\n" will
     not be written.

tests/hard_coded/Mmakefile:
tests/hard_coded/format_term_nl.{exp,m}:
     Add a regression test.
2026-02-20 16:03:18 +11:00

1394 lines
53 KiB
Mathematica

%---------------------------------------------------------------------------%
% vim: ft=mercury ts=4 sw=4 et
%---------------------------------------------------------------------------%
% Copyright (C) 1994-2006, 2009, 2011-2012 The University of Melbourne.
% Copyright (C) 2014-2026 The Mercury team.
% This file is distributed under the terms specified in COPYING.LIB.
%---------------------------------------------------------------------------%
%
% File: term_io.m.
% Main author: fjh.
% Stability: high.
%
% This module provides predicates to write out terms that use the ground
% representation defined in term.m.
%
% Predicates to read in such terms are available in mercury_term_parser.m.
%
% All the operations exported by this module write out a term or a component
% of a term. Each operation is available in four versions.
%
% 1. A function that converts the given term or component to a string.
% 2a. A predicate that writes out the given term or component
% to the current output stream,
% 2b. A predicate that writes out the given term or component
% to a specified output stream,
% 3. A predicate that writes out the given term or component
% to any entity that implements the appropriate stream operations.
%
% These versions normally follow a naming scheme:
%
% - X_to_string for version 1,
% - write_X for version 2a and 2b (these two differing only in arity,
% with 2b having an extra initial output stream argument), and
% - format_X for version 3
%
% where X is the name of the entity being operated on. However, when
% X is a string, the name of the function version will not end in `to_string',
% since that would be strange.
%
% Some operations have more than these four versions, but in every case,
% the extra versions are just old and now obsolete names for one of the four.
%
% The four versions of the same operation will generate the same output,
% they will just put that output in different places.
%
%---------------------------------------------------------------------------%
%---------------------------------------------------------------------------%
:- module term_io.
:- interface.
:- import_module char.
:- import_module io.
:- import_module ops.
:- import_module stream.
:- import_module term.
:- import_module varset.
%---------------------------------------------------------------------------%
% These operations output a term
% - to a string,
% - to the current output stream,
% - to the specified output stream, or
% - to any implementation of the stream typeclass.
%
% They use the Mercury operator table.
%
% They all output variable names as specified by the given varset.
% They write _N for all unnamed variables, with N starting at 0.
:- func term_to_string(varset(T), term(T)) = string.
:- pred write_term(varset(T)::in, term(T)::in, io::di, io::uo) is det.
:- pred write_term(io.text_output_stream::in, varset(T)::in, term(T)::in,
io::di, io::uo) is det.
:- pred format_term(Stream::in, varset(T)::in, term(T)::in,
State::di, State::uo) is det <= stream.writer(Stream, string, State).
%---------------------%
% These operations output a term
% - to a string,
% - to the current output stream,
% - to the specified output stream, or
% - to any implementation of the stream typeclass,
%
% They use the specified operator table.
%
% They all output variable names as specified by the given varset.
% They write _N for all unnamed variables, with N starting at 0.
:- func term_with_op_table_to_string(OpTable, varset(T), term(T))
= string <= op_table(OpTable).
:- pred write_term_with_op_table(OpTable::in,
varset(T)::in, term(T)::in, io::di, io::uo) is det <= op_table(OpTable).
:- pred write_term_with_op_table(io.text_output_stream::in, OpTable::in,
varset(T)::in, term(T)::in, io::di, io::uo) is det <= op_table(OpTable).
:- pred format_term_with_op_table(Stream::in, OpTable::in,
varset(T)::in, term(T)::in, State::di, State::uo) is det
<= (op_table(OpTable), stream.writer(Stream, string, State)).
%---------------------%
% These operations output a term
% - to a string,
% - to the current output stream,
% - to the specified output stream, or
% - to any implementation of the stream typeclass
% followed by a period and a newline.
%
% They use the Mercury operator table.
%
% They all output variable names as specified by the given varset.
% They write _N for all unnamed variables, with N starting at 0.
:- func term_nl_to_string(varset(T), term(T)) = string.
:- pred write_term_nl(varset(T)::in, term(T)::in, io::di, io::uo) is det.
:- pred write_term_nl(io.text_output_stream::in, varset(T)::in, term(T)::in,
io::di, io::uo) is det.
:- pred format_term_nl(Stream::in, varset(T)::in, term(T)::in,
State::di, State::uo) is det <= stream.writer(Stream, string, State).
%---------------------%
% These operations output a term
% - to a string,
% - to the current output stream,
% - to the specified output stream, or
% - to any implementation of the stream typeclass
% followed by a period and a newline.
%
% They use the specified operator table.
%
% They all output variable names as specified by the given varset.
% They write _N for all unnamed variables, with N starting at 0.
:- func term_nl_with_op_table_to_string(OpTable, varset(T), term(T))
= string <= op_table(OpTable).
:- pred write_term_nl_with_op_table(OpTable::in,
varset(T)::in, term(T)::in, io::di, io::uo) is det <= op_table(OpTable).
:- pred write_term_nl_with_op_table(io.text_output_stream::in, OpTable::in,
varset(T)::in, term(T)::in, io::di, io::uo) is det <= op_table(OpTable).
:- pred format_term_nl_with_op_table(Stream::in, OpTable::in,
varset(T)::in, term(T)::in, State::di, State::uo) is det
<= (op_table(OpTable), stream.writer(Stream, string, State)).
%---------------------%
% These operations output either
% - the value of the variable, if it is bound in the given varset, or
% - the name of the variable, if it is not bound in the given varset,
%
% - to a string,
% - to the current output stream,
% - to the specified output stream, or
% - to any implementation of the stream typeclass
% followed by a period and a newline.
%
% They use the Mercury operator table when printing a value.
%
% They all output variable names as specified by the given varset.
% They write _N for all unnamed variables, with N starting at 0.
:- func variable_to_string(varset(T), var(T)) = string.
:- pred write_variable(varset(T)::in, var(T)::in, io::di, io::uo) is det.
:- pred write_variable(io.text_output_stream::in, varset(T)::in, var(T)::in,
io::di, io::uo) is det.
:- pred format_variable(Stream::in, varset(T)::in, var(T)::in,
State::di, State::uo) is det <= stream.writer(Stream, string, State).
%---------------------%
% These operations output either
% - the value of the variable, if it is bound in the given varset, or
% - the name of the variable, if it is not bound in the given varset,
%
% - to a string,
% - to the current output stream,
% - to the specified output stream, or
% - to any implementation of the stream typeclass
% followed by a period and a newline.
%
% They use the specified operator table when printing a value.
%
% They all output variable names as specified by the given varset.
% They write _N for all unnamed variables, with N starting at 0.
:- func variable_with_op_table_to_string(OpTable, varset(T), var(T)) = string
<= op_table(OpTable).
:- pred write_variable_with_op_table(OpTable::in,
varset(T)::in, var(T)::in, io::di, io::uo) is det <= op_table(OpTable).
:- pred write_variable_with_op_table(io.text_output_stream::in, OpTable::in,
varset(T)::in, var(T)::in, io::di, io::uo) is det <= op_table(OpTable).
:- pred format_variable_with_op_table(Stream::in, OpTable::in,
varset(T)::in, var(T)::in, State::di, State::uo) is det
<= (op_table(OpTable), stream.writer(Stream, string, State)).
%---------------------%
% Convert the given constant to a string.
%
:- func format_constant(const) = string.
% NOTE_TO_IMPLEMENTORS OBS :- pragma obsolete(func(format_constant/1), [constant_to_string/1]).
:- func constant_to_string(const) = string.
% Writes a constant (integer, float, string, or atom) to
% the current output stream, or to the specified output stream.
%
:- pred write_constant(const::in, io::di, io::uo) is det.
:- pred write_constant(io.text_output_stream::in, const::in,
io::di, io::uo) is det.
:- pred format_constant(Stream::in, const::in, State::di, State::uo) is det
<= stream.writer(Stream, string, State).
%---------------------%
% Given a string S, return a version of S in which its characters
% are escaped if necessary. Enclose the string in quotes.
%
:- func quoted_atom(string) = string.
% Given a string S, write a version of S in which its characters
% are escaped if necessary. Enclose the string in quotes.
% Write it to the current output stream, or to the specified output stream.
%
:- pred quote_atom(string::in, io::di, io::uo) is det.
:- pred write_quoted_atom(string::in, io::di, io::uo) is det.
:- pred write_quoted_atom(io.text_output_stream::in, string::in,
io::di, io::uo) is det.
:- pred quote_atom(Stream::in, string::in, State::di, State::uo) is det
<= stream.writer(Stream, string, State).
% NOTE_TO_IMPLEMENTORS OBS :- pragma obsolete(pred(quote_atom/4), [format_quoted_atom/4]).
:- pred format_quoted_atom(Stream::in, string::in, State::di, State::uo) is det
<= stream.writer(Stream, string, State).
% UNDOC_PART_START
%---------------------%
% for use by io.m.
:- type adjacent_to_graphic_token
---> maybe_adjacent_to_graphic_token
; not_adjacent_to_graphic_token.
:- func quoted_atom_agt(string, adjacent_to_graphic_token) = string.
:- pred quote_atom_agt(string::in, adjacent_to_graphic_token::in,
io::di, io::uo) is det.
% NOTE_TO_IMPLEMENTORS OBS :- pragma obsolete(pred(quote_atom_agt/4), [write_quoted_atom_agt/4]).
:- pred write_quoted_atom_agt(string::in,
adjacent_to_graphic_token::in, io::di, io::uo) is det.
:- pred write_quoted_atom_agt(io.text_output_stream::in, string::in,
adjacent_to_graphic_token::in, io::di, io::uo) is det.
:- pred quote_atom_agt(Stream::in, string::in,
adjacent_to_graphic_token::in, State::di, State::uo) is det
<= stream.writer(Stream, string, State).
% NOTE_TO_IMPLEMENTORS OBS :- pragma obsolete(pred(quote_atom_agt/5), [format_quoted_atom_agt/5]).
:- pred format_quoted_atom_agt(Stream::in, string::in,
adjacent_to_graphic_token::in, State::di, State::uo) is det
<= stream.writer(Stream, string, State).
% UNDOC_PART_END
%---------------------------------------------------------------------------%
% Given a string S, return a version of S in which its characters
% are escaped if necessary. Do not enclose the string in quotes.
%
:- func escaped_string(string) = string.
% Given a string S, write a version of S in which its characters
% are escaped if necessary. Do not enclose the string in quotes.
% Write it to the current output stream, or to the specified output stream.
%
:- pred write_escaped_string(string::in, io::di, io::uo) is det.
:- pred write_escaped_string(Stream::in, string::in, State::di, State::uo)
is det <= stream.writer(Stream, string, State).
% NOTE_TO_IMPLEMENTORS OBS :- pragma obsolete(pred(write_escaped_string/4), [format_escaped_string/4]).
:- pred format_escaped_string(Stream::in, string::in, State::di, State::uo)
is det <= stream.writer(Stream, string, State).
%---------------------%
% Given a string S, return a version of S, with its characters escaped
% if necessary, in double-quotes.
%
:- func quoted_string(string) = string.
% Given a string S, write a version of S, with its characters escaped
% if necessary, in double-quotes, to the current output stream,
% or to the specified output stream.
%
:- pred quote_string(string::in, io::di, io::uo) is det.
% NOTE_TO_IMPLEMENTORS OBS :- pragma obsolete(pred(quote_string/3), [write_quoted_string/3]).
:- pred write_quoted_string(string::in, io::di, io::uo) is det.
:- pred write_quoted_string(io.text_output_stream::in, string::in,
io::di, io::uo) is det.
:- pred quote_string(Stream::in, string::in, State::di, State::uo) is det
<= stream.writer(Stream, string, State).
% NOTE_TO_IMPLEMENTORS OBS :- pragma obsolete(pred(quote_string/4), [format_quoted_string/4]).
:- pred format_quoted_string(Stream::in, string::in, State::di, State::uo)
is det <= stream.writer(Stream, string, State).
%---------------------%
% Given a character C, return C, escaped if necessary.
% Do not enclose it in single-quotes.
%
:- func escaped_char(char) = string.
% NOTE_TO_IMPLEMENTORS OBS :- pragma obsolete(func(escaped_char/1), [escaped_char_to_string/1]).
:- func escaped_char_to_string(char) = string.
% Given a character C, write C, escaped if necessary,
% and not enclosed in single-quotes, to the current output stream,
% or to the specified output stream.
%
:- pred write_escaped_char(char::in, io::di, io::uo) is det.
:- pred write_escaped_char(Stream::in, char::in, State::di, State::uo) is det
<= stream.writer(Stream, string, State).
% NOTE_TO_IMPLEMENTORS OBS :- pragma obsolete(pred(write_escaped_char/4), [format_escaped_char/4]).
:- pred format_escaped_char(Stream::in, char::in, State::di, State::uo) is det
<= stream.writer(Stream, string, State).
% A reversible version of escaped_char_to_string.
%
:- pred string_is_escaped_char(char, string).
:- mode string_is_escaped_char(in, out) is det.
:- mode string_is_escaped_char(out, in) is semidet.
%---------------------%
% Given a character C, return C, escaped if necessary, in single-quotes.
%
:- func quoted_char(char) = string.
% NOTE_TO_IMPLEMENTORS OBS :- pragma obsolete(func(quoted_char/1), [quoted_char_to_string/1]).
:- func quoted_char_to_string(char) = string.
% Given a character C, write C, escaped if necessary, in single-quotes,
% to the current output stream, or to the specified output stream.
%
:- pred quote_char(char::in, io::di, io::uo) is det.
% NOTE_TO_IMPLEMENTORS OBS :- pragma obsolete(pred(quote_char/3), [write_quoted_char/3]).
:- pred write_quoted_char(char::in, io::di, io::uo) is det.
:- pred write_quoted_char(io.text_output_stream::in, char::in,
io::di, io::uo) is det.
:- pred quote_char(Stream::in, char::in, State::di, State::uo) is det
<= stream.writer(Stream, string, State).
% NOTE_TO_IMPLEMENTORS OBS :- pragma obsolete(pred(quote_char/4), [format_quoted_char/4]).
:- pred format_quoted_char(Stream::in, char::in, State::di, State::uo) is det
<= stream.writer(Stream, string, State).
%---------------------------------------------------------------------------%
%---------------------------------------------------------------------------%
:- implementation.
% Everything below here is not intended to be part of the public interface,
% and will not be included in the Mercury library reference manual.
%---------------------------------------------------------------------------%
:- interface.
:- import_module string.
:- import_module string.builder.
%---------------------------------------------------------------------------%
% Convert `integer_base' constant to its numeric value.
%
:- func integer_base_int(integer_base) = int.
% Return the prefix for integer literals of the given base.
%
:- func integer_base_prefix(integer_base) = string.
% Convert a character to the corresponding octal escape code.
%
% We use ISO-Prolog style octal escapes, which are of the form '\nnn\';
% note that unlike C octal escapes, they are terminated with a backslash.
%
% XXX Using this predicate in the compiler may cause problems interfacing
% with versions of the compiler that have been built in grades which use
% different character representations.
%
:- func mercury_escape_char(char) = string.
% Succeed if the given character is a Mercury punctuation character.
%
:- pred is_mercury_punctuation_char(char::in) is semidet.
% encode_escaped_char(Char, Str):
%
% Succeed in one of two cases:
%
% - Char is 'x', and Str is "x", where x is a valid Mercury source
% character, or
% - Char is '\x' and Str is "\x", where '\x' is a valid character
% escape sequence.
%
:- pred encode_escaped_char(char, string).
:- mode encode_escaped_char(in, out) is semidet.
:- mode encode_escaped_char(out, in) is semidet.
:- pragma type_spec_constrained_preds(
[stream.writer(Stream, string, State)],
do_not_apply_to_superclasses,
[subst([Stream = io.text_output_stream, State = io.state]),
subst([Stream = string.builder.handle, State = string.builder.state])]).
%---------------------------------------------------------------------------%
%---------------------------------------------------------------------------%
:- implementation.
:- import_module bool.
:- import_module counter.
:- import_module int.
:- import_module integer.
:- import_module list.
:- import_module mercury_term_lexer.
:- import_module require.
:- import_module stream.string_writer.
%---------------------------------------------------------------------------%
%---------------------------------------------------------------------------%
term_to_string(VarSet, Term) = Str :-
State0 = string.builder.init,
format_term(string.builder.handle, VarSet, Term, State0, State),
Str = string.builder.to_string(State).
write_term(VarSet, Term, !IO) :-
io.output_stream(OutStream, !IO),
format_term(OutStream, VarSet, Term, !IO).
write_term(OutStream, VarSet, Term, !IO) :-
format_term(OutStream, VarSet, Term, !IO).
format_term(Stream, VarSet, Term, !State) :-
OpTable = init_mercury_op_table,
format_term_with_op_table(Stream, OpTable, VarSet, Term, !State).
%---------------------%
term_with_op_table_to_string(OpTable, VarSet, Term) = Str :-
State0 = string.builder.init,
format_term_with_op_table(string.builder.handle, OpTable, VarSet, Term,
State0, State),
Str = string.builder.to_string(State).
write_term_with_op_table(OpTable, VarSet, Term, !IO) :-
io.output_stream(OutStream, !IO),
format_term_with_op_table(OutStream, OpTable, VarSet, Term, !IO).
write_term_with_op_table(OutStream, OpTable, VarSet, Term, !IO) :-
format_term_anon_vars(OutStream, OpTable, Term, VarSet, _,
anon_var_to_int, _, !IO).
format_term_with_op_table(Stream, OpTable, VarSet, Term, !State) :-
format_term_anon_vars(Stream, OpTable, Term, VarSet, _,
anon_var_to_int, _, !State).
%---------------------%
term_nl_to_string(VarSet, Term) = Str :-
State0 = string.builder.init,
format_term_nl(string.builder.handle, VarSet, Term, State0, State),
Str = string.builder.to_string(State).
write_term_nl(VarSet, Term, !IO) :-
io.output_stream(OutStream, !IO),
write_term_nl(OutStream, VarSet, Term, !IO).
write_term_nl(OutStream, VarSet, Term, !IO) :-
OpTable = init_mercury_op_table,
write_term_nl_with_op_table(OutStream, OpTable, VarSet, Term, !IO).
format_term_nl(Stream, VarSet, Term, !State) :-
OpTable = init_mercury_op_table,
format_term_nl_with_op_table(Stream, OpTable, VarSet, Term, !State).
%---------------------%
term_nl_with_op_table_to_string(OpTable, VarSet, Term) = Str :-
State0 = string.builder.init,
format_term_nl_with_op_table(string.builder.handle, OpTable, VarSet, Term,
State0, State),
Str = string.builder.to_string(State).
write_term_nl_with_op_table(OpTable, VarSet, Term, !IO) :-
io.output_stream(OutStream, !IO),
write_term_nl_with_op_table(OutStream, OpTable, VarSet, Term, !IO).
write_term_nl_with_op_table(OutStream, OpTable, VarSet, Term, !IO) :-
format_term_nl_with_op_table(OutStream, OpTable, VarSet, Term, !IO).
format_term_nl_with_op_table(Stream, OpTable, VarSet, Term, !State) :-
format_term_with_op_table(Stream, OpTable, VarSet, Term, !State),
stream.put(Stream, ".\n", !State).
%---------------------------------------------------------------------------%
:- pred format_term_anon_vars(Stream::in, OpTable::in,
term(T)::in, varset(T)::in, varset(T)::out,
anon_var_info::in, anon_var_info::out, State::di, State::uo) is det
<= (op_table(OpTable), stream.writer(Stream, string, State)).
format_term_anon_vars(Stream, OpTable, Term, !VarSet, !Anon, !State) :-
format_term_prio_anon_vars(Stream, OpTable, Term,
ops.universal_priority(OpTable), !VarSet, !Anon, !State).
:- pred format_term_prio_anon_vars(Stream::in, OpTable::in,
term(T)::in, ops.priority::in, varset(T)::in, varset(T)::out,
anon_var_info::in, anon_var_info::out, State::di, State::uo) is det
<= (op_table(OpTable), stream.writer(Stream, string, State)).
format_term_prio_anon_vars(Stream, OpTable, Term, Priority,
!VarSet, !Anon, !State) :-
(
Term = term.variable(Var, _),
format_variable_anon_vars(Stream, OpTable, Var, !VarSet, !Anon, !State)
;
Term = term.functor(Functor, ArgTerms, _),
(
( Functor = term.integer(_, _, _, _)
; Functor = term.string(_)
; Functor = term.float(_)
; Functor = term.implementation_defined(_)
),
% Terms with these functors should NOT have arguments.
(
ArgTerms = [],
format_constant_agt(Stream, Functor,
maybe_adjacent_to_graphic_token, !State)
;
ArgTerms = [_ | _],
unexpected($pred, "constant has arguments")
)
;
Functor = atom(Atom),
format_atom_term_prio_anon_vars(Stream, OpTable, Atom, ArgTerms,
Priority, !VarSet, !Anon, !State)
)
).
:- pred format_atom_term_prio_anon_vars(Stream::in,
OpTable::in, string::in, list(term(T))::in, ops.priority::in,
varset(T)::in, varset(T)::out, anon_var_info::in, anon_var_info::out,
State::di, State::uo) is det
<= (op_table(OpTable), stream.writer(Stream, string, State)).
format_atom_term_prio_anon_vars(Stream, OpTable, Atom, ArgTerms,
Priority, !VarSet, !Anon, !State) :-
% NOTE It would be nice to handle the first four conditions at once
% with code such as
% ( if
% ( Atom = "[|]"
% ; Atom = "[]"
% ; Atom = "{}"
% ; Atom = ""
% )
% then
% ...
% else
% ...
% )
%
% Unfortunately, three things conspire together to prevent code
% along those lines from being an improvement:
%
% - the fact that the special-case codes for these values of Atom
% apply only if ArgTerms has the right shape,
%
% - the fact that these "right shapes" are different for these
% values of Atom, and
%
% - that the fallback code, which we should execute if ArgTerms
% does NOT have the right shape, is more complicated than a simple call
% to format_atom_term_prio_anon_vars_std, since it must also consider
% the possibility that Atom is an operator in a nonstandard OpTable.
( if
Atom = "[|]",
ArgTerms = [ListHead, ListTail]
then
stream.put(Stream, "[", !State),
format_term_arg(Stream, OpTable, ListHead, !VarSet, !Anon, !State),
format_later_list_elements(Stream, OpTable, ListTail,
!VarSet, !Anon, !State),
stream.put(Stream, "]", !State)
else if
Atom = "[]",
ArgTerms = []
then
stream.put(Stream, "[]", !State)
else if
Atom = "{}",
ArgTerms = [BracedHeadTerm | BracedTailTerms]
then
(
BracedTailTerms = [],
% Add spaces around the one argument term.
stream.put(Stream, "{ ", !State),
format_term_anon_vars(Stream, OpTable, BracedHeadTerm,
!VarSet, !Anon, !State),
stream.put(Stream, " }", !State)
;
BracedTailTerms = [_ | _],
% Do not add spaces around the several argument terms.
stream.put(Stream, "{", !State),
format_term_arg(Stream, OpTable, BracedHeadTerm,
!VarSet, !Anon, !State),
format_term_later_args(Stream, OpTable, BracedTailTerms,
!VarSet, !Anon, !State),
stream.put(Stream, "}", !State)
)
else if
% The empty functor '' is used for higher-order syntax:
% Var(Arg, ...) gets parsed as ''(Var, Arg). When writing it out,
% we want to use the nice syntax.
Atom = "",
ArgTerms = [term.variable(Var, _), FirstArg | OtherArgTerms]
then
format_variable_anon_vars(Stream, OpTable, Var,
!VarSet, !Anon, !State),
stream.put(Stream, "(", !State),
format_term_arg(Stream, OpTable, FirstArg, !VarSet, !Anon, !State),
format_term_later_args(Stream, OpTable, OtherArgTerms,
!VarSet, !Anon, !State),
stream.put(Stream, ")", !State)
else if
lookup_op_infos(OpTable, Atom, OpInfos)
then
OpInfos = op_infos(MaybeInfix, MaybeBinPrefix,
MaybePrefix, MaybePostfix),
(
( ArgTerms = []
; ArgTerms = [_, _, _ | _]
),
format_atom_term_prio_anon_vars_std(Stream, OpTable,
Atom, ArgTerms, Priority, !VarSet, !Anon, !State)
;
ArgTerms = [ArgTerm1],
( if MaybePrefix = pre(OpPriority, OpGtOrGe) then
maybe_write_paren(Stream, "(", Priority, OpPriority, !State),
format_constant_atom(Stream, Atom, !State),
stream.put(Stream, " ", !State),
NewPriority = min_priority_for_arg(OpPriority, OpGtOrGe),
format_term_prio_anon_vars(Stream, OpTable, ArgTerm1,
NewPriority, !VarSet, !Anon, !State),
maybe_write_paren(Stream, ")", Priority, OpPriority, !State)
else if MaybePostfix = post(OpPriority, OpGtOrGe) then
maybe_write_paren(Stream, "(", Priority, OpPriority, !State),
NewPriority = min_priority_for_arg(OpPriority, OpGtOrGe),
format_term_prio_anon_vars(Stream, OpTable, ArgTerm1,
NewPriority, !VarSet, !Anon, !State),
stream.put(Stream, " ", !State),
format_constant_atom(Stream, Atom, !State),
maybe_write_paren(Stream, ")", Priority, OpPriority, !State)
else
format_atom_term_prio_anon_vars_std(Stream, OpTable,
Atom, ArgTerms, Priority, !VarSet, !Anon, !State)
)
;
ArgTerms = [ArgTerm1, ArgTerm2],
( if
MaybeInfix = in(OpPriority, LeftGtOrGe, RightGtOrGe)
then
maybe_write_paren(Stream, "(", Priority, OpPriority, !State),
LeftPriority = min_priority_for_arg(OpPriority, LeftGtOrGe),
format_term_prio_anon_vars(Stream, OpTable, ArgTerm1,
LeftPriority, !VarSet, !Anon, !State),
( if Atom = "," then
stream.put(Stream, ", ", !State)
else if Atom = "." then
% If the operator is '.'/2, then we must not put spaces
% around it (or at the very least, we should not put spaces
% afterwards) because that would make it appear as the
% end-of-term token. However, we do have to quote it
% if the right hand side can begin with a digit.
( if starts_with_digit(ArgTerm2) then
Dot = "'.'"
else
Dot = "."
),
stream.put(Stream, Dot, !State)
else
stream.put(Stream, " ", !State),
format_constant_atom(Stream, Atom, !State),
stream.put(Stream, " ", !State)
),
RightPriority = min_priority_for_arg(OpPriority, RightGtOrGe),
format_term_prio_anon_vars(Stream, OpTable, ArgTerm2,
RightPriority, !VarSet, !Anon, !State),
maybe_write_paren(Stream, ")", Priority, OpPriority, !State)
else if
MaybeBinPrefix = bin_pre(OpPriority, LeftGtOrGe, RightGtOrGe)
then
maybe_write_paren(Stream, "(", Priority, OpPriority, !State),
format_constant_atom(Stream, Atom, !State),
stream.put(Stream, " ", !State),
LeftPriority = min_priority_for_arg(OpPriority, LeftGtOrGe),
format_term_prio_anon_vars(Stream, OpTable, ArgTerm1,
LeftPriority, !VarSet, !Anon, !State),
stream.put(Stream, " ", !State),
RightPriority = min_priority_for_arg(OpPriority, RightGtOrGe),
format_term_prio_anon_vars(Stream, OpTable, ArgTerm2,
RightPriority, !VarSet, !Anon, !State),
maybe_write_paren(Stream, ")", Priority, OpPriority, !State)
else
format_atom_term_prio_anon_vars_std(Stream, OpTable,
Atom, ArgTerms, Priority, !VarSet, !Anon, !State)
)
)
else
format_atom_term_prio_anon_vars_std(Stream, OpTable, Atom, ArgTerms,
Priority, !VarSet, !Anon, !State)
).
:- pred format_atom_term_prio_anon_vars_std(Stream::in,
OpTable::in, string::in, list(term(T))::in, ops.priority::in,
varset(T)::in, varset(T)::out, anon_var_info::in, anon_var_info::out,
State::di, State::uo) is det
<= (op_table(OpTable), stream.writer(Stream, string, State)).
format_atom_term_prio_anon_vars_std(Stream, OpTable, Atom, ArgTerms,
Priority, !VarSet, !Anon, !State) :-
( if
ArgTerms = [],
ops.is_op(OpTable, Atom),
priority_ge(Priority, ops.loosest_op_priority(OpTable))
then
stream.put(Stream, "(", !State),
format_constant_atom(Stream, Atom, !State),
stream.put(Stream, ")", !State)
else
format_constant_atom(Stream, Atom, !State)
),
(
ArgTerms = [HeadArgTerm | TailArgTerms],
stream.put(Stream, "(", !State),
format_term_arg(Stream, OpTable, HeadArgTerm, !VarSet, !Anon, !State),
format_term_later_args(Stream, OpTable, TailArgTerms,
!VarSet, !Anon, !State),
stream.put(Stream, ")", !State)
;
ArgTerms = []
).
:- pred format_term_arg(Stream::in, OpTable::in,
term(T)::in, varset(T)::in, varset(T)::out,
anon_var_info::in, anon_var_info::out, State::di, State::uo) is det
<= (op_table(OpTable), stream.writer(Stream, string, State)).
format_term_arg(Stream, OpTable, Term, !VarSet, !Anon, !State) :-
format_term_prio_anon_vars(Stream, OpTable, Term,
ops.arg_priority(OpTable), !VarSet, !Anon, !State).
% Write the remaining arguments.
%
:- pred format_term_later_args(Stream::in, OpTable::in,
list(term(T))::in, varset(T)::in, varset(T)::out,
anon_var_info::in, anon_var_info::out, State::di, State::uo) is det
<= (op_table(OpTable), stream.writer(Stream, string, State)).
format_term_later_args(_, _, [], !VarSet, !Anon, !State).
format_term_later_args(Stream, OpTable, [X | Xs], !VarSet, !Anon, !State) :-
stream.put(Stream, ", ", !State),
format_term_arg(Stream, OpTable, X, !VarSet, !Anon, !State),
format_term_later_args(Stream, OpTable, Xs, !VarSet, !Anon, !State).
:- pred format_later_list_elements(Stream::in, OpTable::in,
term(T)::in, varset(T)::in, varset(T)::out,
anon_var_info::in, anon_var_info::out, State::di, State::uo) is det
<= (op_table(OpTable), stream.writer(Stream, string, State)).
format_later_list_elements(Stream, OpTable, Term, !VarSet, !Anon, !State) :-
( if
Term = term.variable(Var, _),
varset.search_var(!.VarSet, Var, Value)
then
format_later_list_elements(Stream, OpTable, Value,
!VarSet, !Anon, !State)
else if
Term = term.functor(term.atom("[|]"), [ListHead, ListTail], _)
then
stream.put(Stream, ", ", !State),
format_term_arg(Stream, OpTable, ListHead, !VarSet, !Anon, !State),
format_later_list_elements(Stream, OpTable, ListTail,
!VarSet, !Anon, !State)
else if
Term = term.functor(term.atom("[]"), [], _)
then
true
else
stream.put(Stream, " | ", !State),
format_term_anon_vars(Stream, OpTable, Term, !VarSet, !Anon, !State)
).
% Succeeds if-and-only-if outputting the given term would start
% with a digit.
% (This is a safe, conservative approximation and is used to decide
% whether or not to quote infix '.'/2.)
%
:- pred starts_with_digit(term(T)::in) is semidet.
starts_with_digit(functor(integer(_, _, _, _), _, _)).
starts_with_digit(functor(float(_), _, _)).
starts_with_digit(functor(atom(Op), Args, _)) :-
(
Args = [Arg, _],
ops.lookup_infix_op(ops.init_mercury_op_table, Op, _, _, _)
;
Args = [Arg],
ops.lookup_postfix_op(ops.init_mercury_op_table, Op, _, _)
),
starts_with_digit(Arg).
%---------------------------------------------------------------------------%
variable_to_string(VarSet, Var) = Str :-
State0 = string.builder.init,
format_variable(string.builder.handle, VarSet, Var,
State0, State),
Str = string.builder.to_string(State).
write_variable(VarSet, Var, !IO) :-
io.output_stream(OutStream, !IO),
write_variable(OutStream, VarSet, Var, !IO).
write_variable(OutStream, VarSet, Var, !IO) :-
OpTable = init_mercury_op_table,
term_io.write_variable_with_op_table(OutStream, OpTable, VarSet, Var, !IO).
format_variable(Stream, VarSet, Var, !State) :-
OpTable = init_mercury_op_table,
term_io.format_variable_with_op_table(Stream, OpTable, VarSet, Var,
!State).
%---------------------%
variable_with_op_table_to_string(OpTable, VarSet, Var) = Str :-
State0 = string.builder.init,
format_variable_with_op_table(string.builder.handle, OpTable, VarSet, Var,
State0, State),
Str = string.builder.to_string(State).
write_variable_with_op_table(OpTable, VarSet, Var, !IO) :-
io.output_stream(OutStream, !IO),
write_variable_with_op_table(OutStream, OpTable, VarSet, Var, !IO).
write_variable_with_op_table(OutStream, OpTable, VarSet, Var, !IO) :-
format_variable_anon_vars(OutStream, OpTable, Var, VarSet, _,
anon_var_to_int, _, !IO).
format_variable_with_op_table(Stream, OpTable, VarSet, Var, !State) :-
format_variable_anon_vars(Stream, OpTable, Var, VarSet, _,
anon_var_to_int, _, !State).
%---------------------%
:- type anon_var_info
---> anon_var_to_int
% The string we use to write out an anonymous variable
% should be derived from its variable number, obtained
% by calling var_to_int on it.
; anon_occur_order(counter).
% The string we use to write out an anonymous variable
% should be based on whether it is the first, second, third etc
% anonymous variable that our traversal of the whole term
% has encountered. The counter tells us the number we should give
% to the *next* one we encounter.
% Write a variable.
%
% There are two ways we could choose to write unnamed variables.
%
% 1 Convert the variable to the integer that represents it and write
% `_N' where N is that integer. This has the advantage that
% such variables get printed in a canonical way, so rearranging terms
% containing such variables will not affect the way they are numbered
% (this includes breaking up a term and printing the pieces separately).
% We use this way if !.Anon is anon_var_to_int.
%
% 2 Number the unnamed variables from 0 and write `_N' where
% N is the next number in the sequence of such variables.
% This has the advantage that such variables can be visually scanned
% rather more easily (for example in error messages).
% We use this way if !.Anon is anon_occur_order.
%
% We provide full support for both approaches internally to term_io.m,
% but expose only anon_var_to_int to users, for now.
%
% XXX The names we generate here, with either approach,
% *could* clash with the name of an explicit-named variable.
%
:- pred format_variable_anon_vars(Stream::in, OpTable::in, var(T)::in,
varset(T)::in, varset(T)::out, anon_var_info::in, anon_var_info::out,
State::di, State::uo) is det
<= (op_table(OpTable), stream.writer(Stream, string, State)).
format_variable_anon_vars(Stream, OpTable, Var, !VarSet, !Anon, !State) :-
( if varset.search_var(!.VarSet, Var, Value) then
format_term_anon_vars(Stream, OpTable, Value, !VarSet, !Anon, !State)
else if varset.search_name(!.VarSet, Var, Name) then
stream.put(Stream, Name, !State)
else
(
!.Anon = anon_var_to_int,
term.var_to_int(Var, VarNum)
;
!.Anon = anon_occur_order(Counter0),
counter.allocate(VarNum, Counter0, Counter),
!:Anon = anon_occur_order(Counter)
),
string.format("_%d", [i(VarNum)], VarName),
% Recording the name we have given Var in !VarSet is needed only
% with anon_occur_order; with anon_var_to_int, we would give the same
% name to the same variable even without it. However, that would
% require allocating memory for VarName on *every* use. The alternative
% is recording the name in !VarSet here. Depending on the length
% of VarName and the depth of the var name map in !.VarSet,
% this recording could take as much time and memory as several
% reconstructions of VarName. Since we don't know how many more
% times we will need the name of Var, we cannot know for sure
% which approach will be faster or use more memory. The difference
% if that the memory usage of the "record the name" approach is
% *bounded*, while the memory usage of the "rebuild the name each time"
% approach is not. That is why we choose the former even with
% anon_var_to_int.
varset.name_var(Var, VarName, !VarSet),
stream.put(Stream, VarName, !State)
).
%---------------------------------------------------------------------------%
format_constant(Const) =
constant_to_string(Const).
constant_to_string(Const) = Str :-
AGT = not_adjacent_to_graphic_token,
State0 = string.builder.init,
format_constant_agt(string.builder.handle, Const, AGT, State0, State),
Str = string.builder.to_string(State).
write_constant(Const, !IO) :-
io.output_stream(OutStream, !IO),
write_constant(OutStream, Const, !IO).
write_constant(OutStream, Const, !IO) :-
AGT = not_adjacent_to_graphic_token,
format_constant_agt(OutStream, Const, AGT, !IO).
format_constant(Stream, Const, !State) :-
AGT = not_adjacent_to_graphic_token,
format_constant_agt(Stream, Const, AGT, !State).
:- pred format_constant_agt(Stream::in, const::in,
adjacent_to_graphic_token::in, State::di, State::uo) is det
<= stream.writer(Stream, string, State).
format_constant_agt(Stream, Const, AGT, !State) :-
(
Const = term.integer(Base, Int, Signedness, Size),
Prefix = integer_base_prefix(Base),
IntStr = integer.to_base_string(Int, integer_base_int(Base)),
Suffix = integer_signedness_and_size_suffix(Signedness, Size),
stream.put(Stream, Prefix, !State),
stream.put(Stream, IntStr, !State),
stream.put(Stream, Suffix, !State)
;
Const = term.float(Float),
put_float(Stream, Float, !State)
;
Const = term.atom(Atom),
term_io.format_quoted_atom_agt(Stream, Atom, AGT, !State)
;
Const = term.string(Str),
term_io.format_quoted_string(Stream, Str, !State)
;
Const = term.implementation_defined(ImplDef),
stream.put(Stream, "$", !State),
put(Stream, ImplDef, !State)
).
% This does the job of format_constant just for atom constants.
%
:- pred format_constant_atom(Stream::in, string::in, State::di, State::uo)
is det <= stream.writer(Stream, string, State).
:- pragma inline(pred(format_constant_atom/4)).
format_constant_atom(Stream, Atom, !IO) :-
AGT = not_adjacent_to_graphic_token,
term_io.format_quoted_atom_agt(Stream, Atom, AGT, !IO).
:- func integer_signedness_and_size_suffix(term.signedness,
term.integer_size) = string.
integer_signedness_and_size_suffix(signed, size_word) = "".
integer_signedness_and_size_suffix(signed, size_8_bit) = "i8".
integer_signedness_and_size_suffix(signed, size_16_bit) = "i16".
integer_signedness_and_size_suffix(signed, size_32_bit) = "i32".
integer_signedness_and_size_suffix(signed, size_64_bit) = "i64".
integer_signedness_and_size_suffix(unsigned, size_word) = "u".
integer_signedness_and_size_suffix(unsigned, size_8_bit) = "u8".
integer_signedness_and_size_suffix(unsigned, size_16_bit) = "u16".
integer_signedness_and_size_suffix(unsigned, size_32_bit) = "u32".
integer_signedness_and_size_suffix(unsigned, size_64_bit) = "u64".
%---------------------------------------------------------------------------%
quoted_atom(Str0) = Str :-
State0 = string.builder.init,
format_quoted_atom(string.builder.handle, Str0, State0, State),
Str = string.builder.to_string(State).
quote_atom(Str, !IO) :-
term_io.write_quoted_atom(Str, !IO).
write_quoted_atom(Str, !IO) :-
AGT = not_adjacent_to_graphic_token,
term_io.write_quoted_atom_agt(Str, AGT, !IO).
write_quoted_atom(OutStream, Str, !IO) :-
AGT = not_adjacent_to_graphic_token,
term_io.write_quoted_atom_agt(OutStream, Str, AGT, !IO).
quote_atom(Stream, Str, !State) :-
format_quoted_atom(Stream, Str, !State).
format_quoted_atom(Stream, Str, !State) :-
AGT = not_adjacent_to_graphic_token,
term_io.format_quoted_atom_agt(Stream, Str, AGT, !State).
%---------------------%
quoted_atom_agt(Str0, AGT) = Str :-
State0 = string.builder.init,
format_quoted_atom_agt(string.builder.handle, Str0, AGT, State0, State),
Str = string.builder.to_string(State).
quote_atom_agt(Str, AGT, !IO) :-
io.output_stream(Stream, !IO),
term_io.write_quoted_atom_agt(Stream, Str, AGT, !IO).
write_quoted_atom_agt(Str, AGT, !IO) :-
io.output_stream(OutStream, !IO),
write_quoted_atom_agt(OutStream, Str, AGT, !IO).
write_quoted_atom_agt(OutStream, Str, AGT, !IO) :-
format_quoted_atom_agt(OutStream, Str, AGT, !IO).
quote_atom_agt(Stream, Str, AGT, !State) :-
term_io.format_quoted_atom_agt(Stream, Str, AGT, !State).
format_quoted_atom_agt(Stream, Str, AGT, !State) :-
ShouldQuote = should_atom_be_quoted(Str, AGT),
(
ShouldQuote = no,
stream.put(Stream, Str, !State)
;
ShouldQuote = yes,
stream.put(Stream, "'", !State),
term_io.format_escaped_string(Stream, Str, !State),
stream.put(Stream, "'", !State)
).
:- func should_atom_be_quoted(string, adjacent_to_graphic_token) = bool.
should_atom_be_quoted(S, AdjacentToGraphicToken) = ShouldQuote :-
( if
% I didn't make these rules up: see ISO Prolog 6.3.1.3 and 6.4.2. -fjh
(
% Letter digit token (6.4.2)
string.index(S, 0, FirstChar),
char.is_lower(FirstChar),
string.is_all_alnum_or_underscore(S)
;
% Semicolon token (6.4.2)
S = ";"
;
% Cut token (6.4.2)
S = "!"
;
% Graphic token (6.4.2)
string.all_match(mercury_term_lexer.graphic_token_char, S),
S \= "",
% We need to quote tokens starting with '#', because Mercury uses
% '#' to start source line number indicators.
not string.index(S, 0, '#'),
% If the token could be the last token in a term, and the term
% could be followed with ".\n", then we need to quote the token,
% otherwise the "." would be considered part of the same graphic
% token. We can only leave it unquoted if we're sure it won't be
% adjacent to any graphic token.
AdjacentToGraphicToken = not_adjacent_to_graphic_token
;
% 6.3.1.3: atom = open list, close list ;
S = "[]"
;
% 6.3.1.3: atom = open curly, close curly ;
S = "{}"
)
then
ShouldQuote = no
else
% Anything else must be output as a quoted token (6.4.2).
ShouldQuote = yes
).
%---------------------------------------------------------------------------%
escaped_string(Str0) = Str :-
State0 = string.builder.init,
format_escaped_string(string.builder.handle, Str0, State0, State),
Str = string.builder.to_string(State).
write_escaped_string(Str, !IO) :-
io.output_stream(Stream, !IO),
term_io.format_escaped_string(Stream, Str, !IO).
write_escaped_string(Stream, Str, !State) :-
format_escaped_string(Stream, Str, !State).
format_escaped_string(Stream, Str, !State) :-
% XXX ILSEQ Decide what to do with ill-formed sequences.
string.foldl(term_io.format_escaped_char(Stream), Str, !State).
%---------------------%
% Note: the code here is similar to code in compiler/parse_tree_out_pragma.m;
% any changes here may require similar changes there.
quoted_string(Str0) = Str :-
State0 = string.builder.init,
format_quoted_string(string.builder.handle, Str0, State0, State),
Str = string.builder.to_string(State).
quote_string(Str, !IO) :-
io.output_stream(OutStream, !IO),
term_io.format_quoted_string(OutStream, Str, !IO).
write_quoted_string(Str, !IO) :-
io.output_stream(OutStream, !IO),
format_quoted_string(OutStream, Str, !IO).
write_quoted_string(OutStream, Str, !State) :-
format_quoted_string(OutStream, Str, !State).
quote_string(Stream, Str, !State) :-
format_quoted_string(Stream, Str, !State).
format_quoted_string(Stream, Str, !State) :-
stream.put(Stream, "\"", !State),
term_io.format_escaped_string(Stream, Str, !State),
stream.put(Stream, "\"", !State).
%---------------------------------------------------------------------------%
escaped_char(Char) =
escaped_char_to_string(Char).
escaped_char_to_string(Char) = String :-
string_is_escaped_char(Char, String).
write_escaped_char(Char, !IO) :-
io.output_stream(Stream, !IO),
term_io.format_escaped_char(Stream, Char, !IO).
write_escaped_char(Stream, Char, !State) :-
format_escaped_char(Stream, Char, !State).
format_escaped_char(Stream, Char, !State) :-
% Note: the code of format_escaped_char is similar to code in
% compiler/parse_tree_out_pragma.m and MR_escape_string_quote
% in runtime/mercury_string.c; any changes here may require similar
% changes in those spots.
( if mercury_escape_special_char(Char, QuoteChar) then
stream.put(Stream, "\\", !State),
stream.put(Stream, char_to_string(QuoteChar), !State)
else if is_mercury_source_char(Char) then
stream.put(Stream, char_to_string(Char), !State)
else
stream.put(Stream, mercury_escape_char(Char), !State)
).
:- pragma promise_equivalent_clauses(pred(string_is_escaped_char/2)).
string_is_escaped_char(Char::in, String::out) :-
( if mercury_escape_special_char(Char, QuoteChar) then
String = string.append("\\", string.char_to_string(QuoteChar))
else if is_mercury_source_char(Char) then
String = string.char_to_string(Char)
else
String = mercury_escape_char(Char)
).
string_is_escaped_char(Char::out, String::in) :-
% XXX ILSEQ Decide what to do with ill-formed sequences.
string.to_char_list(String, Chars),
(
Chars = [Char],
( is_mercury_source_char(Char)
; mercury_escape_special_char(Char, _QuoteChar)
)
;
Chars = ['\\', QuoteChar],
mercury_escape_special_char(Char, QuoteChar)
;
Chars = ['\\', Char1, Char2, Char3],
NumChars = [Char1, Char2, Char3],
string.from_char_list(NumChars, NumString),
string.base_string_to_int(8, NumString, Int),
char.to_int(Char, Int)
).
% Succeed if Char is a character which is allowed in Mercury string
% and character literals.
%
% Note: the code here is similar to code in the following spots:
%
% - compiler/parse_tree_out_pragma.m.
% - runtime/mercury_trace_base.c
%
% Any changes here may require similar changes there.
%
:- pred is_mercury_source_char(char::in) is semidet.
is_mercury_source_char(Char) :-
( char.is_alnum(Char)
; is_mercury_punctuation_char(Char)
; char.to_int(Char) >= 0xA0 % 0x7f - 0x9f are control characters.
).
%---------------------%
quoted_char(C) =
quoted_char_to_string(C).
quoted_char_to_string(C) =
string.format("'%s'", [s(term_io.escaped_char_to_string(C))]).
quote_char(C, !IO) :-
write_quoted_char(C, !IO).
write_quoted_char(C, !IO) :-
io.output_stream(OutStream, !IO),
format_quoted_char(OutStream, C, !IO).
write_quoted_char(OutStream, C, !IO) :-
format_quoted_char(OutStream, C, !IO).
quote_char(Stream, C, !State) :-
format_quoted_char(Stream, C, !State).
format_quoted_char(Stream, C, !State) :-
stream.put(Stream, term_io.quoted_char_to_string(C), !State).
%---------------------------------------------------------------------------%
%---------------------------------------------------------------------------%
integer_base_int(base_2) = 2.
integer_base_int(base_8) = 8.
integer_base_int(base_10) = 10.
integer_base_int(base_16) = 16.
integer_base_prefix(base_2) = "0b".
integer_base_prefix(base_8) = "0o".
integer_base_prefix(base_10) = "".
integer_base_prefix(base_16) = "0x".
%---------------------------------------------------------------------------%
mercury_escape_char(Char) = EscapeCode :-
char.to_int(Char, Int),
string.int_to_base_string(Int, 8, OctalString0),
string.pad_left(OctalString0, '0', 3, OctalString),
EscapeCode = "\\" ++ OctalString ++ "\\".
%---------------------------------------------------------------------------%
% Currently we only allow the following characters.
% XXX should we just use is_printable(Char) instead?
%
% Note: the code here is similar to code in runtime/mercury_trace_base.c;
% any changes here may require similar changes there.
% Codepoints in 0x20..0x2f.
is_mercury_punctuation_char(' ').
is_mercury_punctuation_char('!').
is_mercury_punctuation_char('"').
is_mercury_punctuation_char('#').
is_mercury_punctuation_char('$').
is_mercury_punctuation_char('%').
is_mercury_punctuation_char('&').
is_mercury_punctuation_char('''').
is_mercury_punctuation_char('(').
is_mercury_punctuation_char(')').
is_mercury_punctuation_char('*').
is_mercury_punctuation_char('+').
is_mercury_punctuation_char(',').
is_mercury_punctuation_char('-').
is_mercury_punctuation_char('.').
is_mercury_punctuation_char('/').
% Codepoints in 0x3a..0x40.
is_mercury_punctuation_char(':').
is_mercury_punctuation_char(';').
is_mercury_punctuation_char('<').
is_mercury_punctuation_char('=').
is_mercury_punctuation_char('>').
is_mercury_punctuation_char('?').
is_mercury_punctuation_char('@').
% Codepoints in 0x5b..0x60.
is_mercury_punctuation_char('[').
is_mercury_punctuation_char('\\').
is_mercury_punctuation_char(']').
is_mercury_punctuation_char('^').
is_mercury_punctuation_char('_').
is_mercury_punctuation_char('`').
% Codepoints in 0x7b..0x7e.
is_mercury_punctuation_char('{').
is_mercury_punctuation_char('|').
is_mercury_punctuation_char('~').
is_mercury_punctuation_char('}').
%---------------------------------------------------------------------------%
%---------------------------------------------------------------------------%
:- pragma promise_equivalent_clauses(pred(encode_escaped_char/2)).
encode_escaped_char(Char::in, Str::out) :-
( if mercury_escape_special_char(Char, EscapeChar) then
string.from_char_list(['\\', EscapeChar], Str)
else if is_mercury_source_char(Char) then
string.from_char_list([Char], Str)
else
fail
).
encode_escaped_char(Char::out, Str::in) :-
% XXX ILSEQ Decide what to do with ill-formed sequences.
string.to_char_list(Str, Chars),
(
Chars = [Char]
;
Chars = ['\\', EscapedChar],
mercury_escape_special_char(Char, EscapedChar)
).
% mercury_escape_special_char(Char, EscapeChar) is true if-and-only-if
% Char is character for which there is a special backslash-escape character
% EscapeChar that can be used after a backslash in string literals or
% atoms to represent Char.
%
% Note: the code here is similar (but not identical) to escape_char
% in mercury_term_lexer.m, and the places listed there.
% Any changes here may require similar changes there.
%
:- pred mercury_escape_special_char(char, char).
:- mode mercury_escape_special_char(in, out) is semidet.
:- mode mercury_escape_special_char(out, in) is semidet.
mercury_escape_special_char('\a', 'a').
mercury_escape_special_char('\b', 'b').
mercury_escape_special_char('\e', 'e').
mercury_escape_special_char('\f', 'f').
mercury_escape_special_char('\n', 'n').
mercury_escape_special_char('\r', 'r').
mercury_escape_special_char('\t', 't').
mercury_escape_special_char('\v', 'v').
mercury_escape_special_char('\\', '\\').
mercury_escape_special_char('''', '''').
mercury_escape_special_char('"', '"').
%---------------------------------------------------------------------------%
:- end_module term_io.
%---------------------------------------------------------------------------%