mirror of
https://github.com/Mercury-Language/mercury.git
synced 2025-12-16 22:35:41 +00:00
Estimated hours taken: 10 Branches: main Split the existing browser library into two libraries, by making the program_representation module into its own library. This is useful because the compiler refers to program_representation.m, whose code thus needs to be linked into compiler executables even if the compiler isn't compiled with debugging enabled. By creating a new library for this module, we avoid any chance of the linker dragging in the rest of the modules in the browser library. (This is a problem with an upcoming diff.). The name of the new library is "mdbcomp", because the intention is that it contain code that is shared between the debugger and the compiler. This means mostly the definitions of data structures that the compiler generates for the debugger, and the predicates that operate on them. Mmake.common.in: Allow MDB_COMP_ as a prefix for symbol names in the browser directory. Mmake.workspace: Add a make variable holding for the name of the new library, and add the name to the relevant lists of libraries. Avoid duplicating the lists of filenames that need to be updated when adding new libraries or changing their names. Mmakefile: Use make variables to refer to library names. browser/mdbcomp.m: browser/mer_mdbcomp.m: Add these files as the top modules of the new library. browser/program_representation.m: Make program_representation.m a submodule of mdbcomp, not mdb. browser/program_representation.m: browser/browser_info.m: Move a predicate from program_representation.m to browser_info.m to avoid the mdbcomp library depend on the browser library, since this would negate the point of the exercise. browser/mdb.m: Delete program_representation.m from the list of submodules. browser/Mmakefile: Update this file to handle the new module. browser/Mercury.options: Mention the new module. browser/*.m: Update the lists of imported modules. Import only one browser module per line. compiler/notes/overall_design.html: Document the new library. compiler/compile_target_code.m: Add the mdbcomp library to the list of libraries we need to link with. compiler/prog_rep.m: trace/mercury_trace_internal.c: Import program_representation.m by its new name. scripts/c2init.in: Centralize knowledge about which files need to be updated when the list of libraries changes here. scripts/c2init.in: scripts/ml.in: tools/binary: tools/binary_step: tools/bootcheck: tools/linear: tools/lml: Update the list of libraries programs are linked with.
738 lines
24 KiB
Mathematica
738 lines
24 KiB
Mathematica
%-----------------------------------------------------------------------------%
|
|
% Copyright (C) 1999-2003 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: declarative_user.m
|
|
% Author: Mark Brown
|
|
% Purpose:
|
|
% This module performs all the user interaction of the front
|
|
% end of the declarative debugger. It is responsible for displaying
|
|
% questions and bugs in a human-readable format, and for getting
|
|
% responses to debugger queries from the user.
|
|
%
|
|
|
|
:- module mdb__declarative_user.
|
|
|
|
:- interface.
|
|
|
|
:- import_module mdb__declarative_debugger.
|
|
|
|
:- import_module list, io.
|
|
|
|
:- type user_question(T)
|
|
---> plain_question(decl_question(T))
|
|
; question_with_default(decl_question(T), decl_truth).
|
|
|
|
:- type user_response(T)
|
|
---> user_answer(decl_question(T), decl_answer(T))
|
|
; no_user_answer
|
|
; exit_diagnosis(T)
|
|
; abort_diagnosis.
|
|
|
|
:- type user_state.
|
|
|
|
:- pred user_state_init(io__input_stream, io__output_stream, user_state).
|
|
:- mode user_state_init(in, in, out) is det.
|
|
|
|
% This predicate handles the interactive part of the declarative
|
|
% debugging process. The user is presented with a question,
|
|
% possibly with a default answer, and is asked to respond about the
|
|
% truth of it in the intended interpretation.
|
|
%
|
|
:- pred query_user(list(user_question(T))::in, user_response(T)::out,
|
|
user_state::in, user_state::out, io__state::di, io__state::uo)
|
|
is cc_multi.
|
|
|
|
% Confirm that the node found is indeed an e_bug or an i_bug.
|
|
%
|
|
:- pred user_confirm_bug(decl_bug::in, decl_confirmation::out,
|
|
user_state::in, user_state::out, io__state::di, io__state::uo)
|
|
is cc_multi.
|
|
|
|
%-----------------------------------------------------------------------------%
|
|
|
|
:- implementation.
|
|
|
|
:- import_module mdb__browser_info.
|
|
:- import_module mdb__browse.
|
|
:- import_module mdb__io_action.
|
|
:- import_module mdb__util.
|
|
:- import_module mdb__declarative_execution.
|
|
:- import_module mdbcomp__program_representation.
|
|
|
|
:- import_module std_util, char, string, bool, int, deconstruct.
|
|
|
|
:- type user_state
|
|
---> user(
|
|
instr :: io__input_stream,
|
|
outstr :: io__output_stream,
|
|
browser :: browser_persistent_state
|
|
).
|
|
|
|
user_state_init(InStr, OutStr, User) :-
|
|
browser_info__init_persistent_state(Browser),
|
|
User = user(InStr, OutStr, Browser).
|
|
|
|
%-----------------------------------------------------------------------------%
|
|
|
|
query_user(Questions, Response, User0, User) -->
|
|
query_user_2(Questions, [], Response, User0, User).
|
|
|
|
:- pred query_user_2(list(user_question(T))::in, list(user_question(T))::in,
|
|
user_response(T)::out, user_state::in, user_state::out,
|
|
io__state::di, io__state::uo) is cc_multi.
|
|
|
|
query_user_2([], _, no_user_answer, User, User) -->
|
|
[].
|
|
query_user_2([UserQuestion | UserQuestions], Skipped, Response, User0, User) -->
|
|
{ Question = get_decl_question(UserQuestion) },
|
|
write_decl_question(Question, User0),
|
|
{ user_question_prompt(UserQuestion, Prompt) },
|
|
get_command(Prompt, Command, User0, User1),
|
|
handle_command(Command, UserQuestion, UserQuestions, Skipped, Response,
|
|
User1, User).
|
|
|
|
:- pred handle_command(user_command::in, user_question(T)::in,
|
|
list(user_question(T))::in, list(user_question(T))::in,
|
|
user_response(T)::out, user_state::in, user_state::out,
|
|
io__state::di, io__state::uo) is cc_multi.
|
|
|
|
handle_command(yes, UserQuestion, _, _, Response, User, User) -->
|
|
{ Question = get_decl_question(UserQuestion) },
|
|
{ Node = get_decl_question_node(Question) },
|
|
{ Response = user_answer(Question, truth_value(Node, yes)) }.
|
|
|
|
handle_command(no, UserQuestion, _, _, Response, User, User) -->
|
|
{ Question = get_decl_question(UserQuestion) },
|
|
{ Node = get_decl_question_node(Question) },
|
|
{ Response = user_answer(Question, truth_value(Node, no)) }.
|
|
|
|
handle_command(inadmissible, UserQuestion, UserQuestions, Skipped, Response,
|
|
User0, User) -->
|
|
io__write_string("Sorry, not implemented,\n"),
|
|
query_user_2([UserQuestion | UserQuestions], Skipped, Response, User0,
|
|
User).
|
|
|
|
handle_command(skip, UserQuestion, UserQuestions, Skipped, Response,
|
|
User0, User) -->
|
|
query_user_2(UserQuestions, [UserQuestion | Skipped], Response, User0,
|
|
User).
|
|
|
|
handle_command(restart, UserQuestion, UserQuestions, Skipped, Response,
|
|
User0, User) -->
|
|
{ reverse_and_append(Skipped, [UserQuestion | UserQuestions],
|
|
RestartedQuestions) },
|
|
query_user(RestartedQuestions, Response, User0, User).
|
|
|
|
handle_command(browse_arg(ArgNum), UserQuestion, UserQuestions, Skipped,
|
|
Response, User0, User) -->
|
|
{ Question = get_decl_question(UserQuestion) },
|
|
{ edt_node_trace_atom(Question, TraceAtom) },
|
|
browse_atom_argument(TraceAtom, ArgNum, MaybeMark, User0, User1),
|
|
(
|
|
{ MaybeMark = no },
|
|
query_user_2([UserQuestion | UserQuestions], Skipped, Response,
|
|
User1, User)
|
|
;
|
|
{ MaybeMark = yes(Mark) },
|
|
{ Which = chosen_head_vars_presentation },
|
|
{
|
|
Which = only_user_headvars,
|
|
ArgPos = user_head_var(ArgNum)
|
|
;
|
|
Which = all_headvars,
|
|
ArgPos = any_head_var(ArgNum)
|
|
},
|
|
{ Node = get_decl_question_node(Question) },
|
|
{ Answer = suspicious_subterm(Node, ArgPos, Mark) },
|
|
{ Response = user_answer(Question, Answer) },
|
|
{ User = User1 }
|
|
).
|
|
|
|
handle_command(print_arg(From, To), UserQuestion, UserQuestions, Skipped,
|
|
Response, User0, User) -->
|
|
{ Question = get_decl_question(UserQuestion) },
|
|
{ edt_node_trace_atom(Question, TraceAtom) },
|
|
print_atom_arguments(TraceAtom, From, To, User0),
|
|
query_user_2([UserQuestion | UserQuestions], Skipped, Response,
|
|
User0, User).
|
|
|
|
handle_command(browse_io(ActionNum), UserQuestion, UserQuestions, Skipped,
|
|
Response, User0, User) -->
|
|
{ Question = get_decl_question(UserQuestion) },
|
|
{ edt_node_io_actions(Question, IoActions) },
|
|
% We don't have code yet to trace a marked I/O action.
|
|
browse_chosen_io_action(IoActions, ActionNum, _MaybeMark, User0, User1),
|
|
query_user_2([UserQuestion | UserQuestions], Skipped, Response,
|
|
User1, User).
|
|
|
|
handle_command(print_io(From, To), UserQuestion, UserQuestions, Skipped,
|
|
Response, User0, User) -->
|
|
{ Question = get_decl_question(UserQuestion) },
|
|
{ edt_node_io_actions(Question, IoActions) },
|
|
print_chosen_io_actions(IoActions, From, To, User0),
|
|
query_user_2([UserQuestion | UserQuestions], Skipped, Response,
|
|
User0, User).
|
|
|
|
handle_command(pd, UserQuestion, _, _, Response, User, User) -->
|
|
{ Question = get_decl_question(UserQuestion) },
|
|
{ Node = get_decl_question_node(Question) },
|
|
{ Response = exit_diagnosis(Node) }.
|
|
|
|
handle_command(abort, _, _, _, Response, User, User) -->
|
|
{ Response = abort_diagnosis }.
|
|
|
|
handle_command(help, UserQuestion, UserQuestions, Skipped, Response,
|
|
User0, User) -->
|
|
user_help_message(User0),
|
|
query_user_2([UserQuestion | UserQuestions], Skipped, Response,
|
|
User0, User).
|
|
|
|
handle_command(empty_command, UserQuestion, UserQuestions, Skipped, Response,
|
|
User0, User) -->
|
|
{
|
|
UserQuestion = plain_question(_),
|
|
Command = skip
|
|
;
|
|
UserQuestion = question_with_default(_, Truth),
|
|
(
|
|
Truth = yes,
|
|
Command = yes
|
|
;
|
|
Truth = no,
|
|
Command = no
|
|
)
|
|
},
|
|
handle_command(Command, UserQuestion, UserQuestions, Skipped, Response,
|
|
User0, User).
|
|
|
|
handle_command(illegal_command, UserQuestion, UserQuestions, Skipped, Response,
|
|
User0, User) -->
|
|
io__write_string("Unknown command, 'h' for help.\n"),
|
|
query_user_2([UserQuestion | UserQuestions], Skipped, Response,
|
|
User0, User).
|
|
|
|
:- func get_decl_question(user_question(T)) = decl_question(T).
|
|
|
|
get_decl_question(plain_question(Q)) = Q.
|
|
get_decl_question(question_with_default(Q, _)) = Q.
|
|
|
|
:- pred user_question_prompt(user_question(T), string).
|
|
:- mode user_question_prompt(in, out) is det.
|
|
|
|
user_question_prompt(plain_question(Question), Prompt) :-
|
|
decl_question_prompt(Question, Prompt).
|
|
|
|
user_question_prompt(question_with_default(Question, DefaultTruth), Prompt) :-
|
|
decl_question_prompt(Question, QuestionPrompt),
|
|
default_prompt(DefaultTruth, DefaultPrompt),
|
|
string__append(QuestionPrompt, DefaultPrompt, Prompt).
|
|
|
|
:- pred decl_question_prompt(decl_question(T), string).
|
|
:- mode decl_question_prompt(in, out) is det.
|
|
|
|
decl_question_prompt(wrong_answer(_, _), "Valid? ").
|
|
decl_question_prompt(missing_answer(_, _, _), "Complete? ").
|
|
decl_question_prompt(unexpected_exception(_, _, _), "Expected? ").
|
|
|
|
:- pred default_prompt(decl_truth, string).
|
|
:- mode default_prompt(in, out) is det.
|
|
|
|
default_prompt(yes, "[yes] ").
|
|
default_prompt(no, "[no] ").
|
|
|
|
:- pred edt_node_trace_atom(decl_question(T)::in, trace_atom::out) is det.
|
|
|
|
edt_node_trace_atom(wrong_answer(_, FinalDeclAtom),
|
|
FinalDeclAtom ^ final_atom).
|
|
edt_node_trace_atom(missing_answer(_, InitDeclAtom, _),
|
|
InitDeclAtom ^ init_atom).
|
|
edt_node_trace_atom(unexpected_exception(_, InitDeclAtom, _),
|
|
InitDeclAtom ^ init_atom).
|
|
|
|
:- pred edt_node_io_actions(decl_question(T)::in, list(io_action)::out) is det.
|
|
|
|
edt_node_io_actions(wrong_answer(_, FinalDeclAtom),
|
|
FinalDeclAtom ^ final_io_actions).
|
|
edt_node_io_actions(missing_answer(_, _, _), []).
|
|
edt_node_io_actions(unexpected_exception(_, _, _), []).
|
|
|
|
:- pred decl_bug_trace_atom(decl_bug::in, trace_atom::out) is det.
|
|
|
|
decl_bug_trace_atom(e_bug(incorrect_contour(FinalDeclAtom, _, _)),
|
|
FinalDeclAtom ^ final_atom).
|
|
decl_bug_trace_atom(e_bug(partially_uncovered_atom(InitDeclAtom, _)),
|
|
InitDeclAtom ^ init_atom).
|
|
decl_bug_trace_atom(e_bug(unhandled_exception(InitDeclAtom, _, _)),
|
|
InitDeclAtom ^ init_atom).
|
|
decl_bug_trace_atom(i_bug(inadmissible_call(_, _, InitDeclAtom, _)),
|
|
InitDeclAtom ^ init_atom).
|
|
|
|
:- pred decl_bug_io_actions(decl_bug::in, list(io_action)::out) is det.
|
|
|
|
decl_bug_io_actions(e_bug(incorrect_contour(FinalDeclAtom, _, _)),
|
|
FinalDeclAtom ^ final_io_actions).
|
|
decl_bug_io_actions(e_bug(partially_uncovered_atom(_, _)), []).
|
|
decl_bug_io_actions(e_bug(unhandled_exception(_, _, _)), []).
|
|
decl_bug_io_actions(i_bug(inadmissible_call(_, _, _, _)), []).
|
|
|
|
:- pred browse_chosen_io_action(list(io_action)::in, int::in,
|
|
maybe(term_path)::out, user_state::in, user_state::out,
|
|
io__state::di, io__state::uo) is cc_multi.
|
|
|
|
browse_chosen_io_action(IoActions, ActionNum, MaybeMark, User0, User) -->
|
|
( { list__index1(IoActions, ActionNum, IoAction) } ->
|
|
browse_io_action(IoAction, MaybeMark, User0, User)
|
|
;
|
|
io__write_string("No such IO action.\n"),
|
|
{ MaybeMark = no },
|
|
{ User = User0 }
|
|
).
|
|
|
|
:- pred print_chosen_io_actions(list(io_action)::in, int::in, int::in,
|
|
user_state::in, io__state::di, io__state::uo) is cc_multi.
|
|
|
|
print_chosen_io_actions(Atom, From, To, User0) -->
|
|
print_chosen_io_action(Atom, From, User0, OK),
|
|
( { OK = yes, From + 1 =< To } ->
|
|
print_chosen_io_actions(Atom, From + 1, To, User0)
|
|
;
|
|
[]
|
|
).
|
|
|
|
:- pred print_chosen_io_action(list(io_action)::in, int::in, user_state::in,
|
|
bool::out, io__state::di, io__state::uo) is cc_multi.
|
|
|
|
print_chosen_io_action(IoActions, ActionNum, User0, OK) -->
|
|
( { list__index1(IoActions, ActionNum, IoAction) } ->
|
|
print_io_action(User0, IoAction),
|
|
{ OK = yes }
|
|
;
|
|
io__write_string("No such IO action.\n"),
|
|
{ OK = no }
|
|
).
|
|
|
|
:- pred browse_io_action(io_action::in, maybe(term_path)::out,
|
|
user_state::in, user_state::out, io__state::di, io__state::uo)
|
|
is cc_multi.
|
|
|
|
browse_io_action(IoAction, MaybeMark, User0, User) -->
|
|
{ io_action_to_synthetic_term(IoAction, ProcName, Args, IsFunc) },
|
|
browse_synthetic(ProcName, Args, IsFunc, User0 ^ instr, User0 ^ outstr,
|
|
MaybeDirs, User0 ^ browser, Browser),
|
|
{ maybe_convert_dirs_to_path(MaybeDirs, MaybeMark) },
|
|
{ User = User0 ^ browser := Browser }.
|
|
|
|
:- pred browse_decl_bug_arg(decl_bug::in, int::in,
|
|
user_state::in, user_state::out, io__state::di, io__state::uo)
|
|
is cc_multi.
|
|
|
|
browse_decl_bug_arg(Bug, ArgNum, User0, User) -->
|
|
{ decl_bug_trace_atom(Bug, Atom) },
|
|
browse_atom_argument(Atom, ArgNum, _, User0, User).
|
|
|
|
:- pred browse_atom_argument(trace_atom::in, int::in, maybe(term_path)::out,
|
|
user_state::in, user_state::out, io__state::di, io__state::uo)
|
|
is cc_multi.
|
|
|
|
browse_atom_argument(Atom, ArgNum, MaybeMark, User0, User) -->
|
|
{ Atom = atom(_, _, Args0) },
|
|
{ maybe_filter_headvars(chosen_head_vars_presentation, Args0, Args) },
|
|
(
|
|
{ list__index1(Args, ArgNum, ArgInfo) },
|
|
{ ArgInfo = arg_info(_, _, MaybeArg) },
|
|
{ MaybeArg = yes(Arg) }
|
|
->
|
|
browse(univ_value(Arg), User0 ^ instr, User0 ^ outstr,
|
|
MaybeDirs, User0 ^ browser, Browser),
|
|
{ maybe_convert_dirs_to_path(MaybeDirs, MaybeMark) },
|
|
{ User = User0 ^ browser := Browser }
|
|
;
|
|
io__write_string(User ^ outstr, "Invalid argument number\n"),
|
|
{ MaybeMark = no },
|
|
{ User = User0 }
|
|
).
|
|
|
|
:- pred print_atom_arguments(trace_atom::in, int::in, int::in, user_state::in,
|
|
io__state::di, io__state::uo) is cc_multi.
|
|
|
|
print_atom_arguments(Atom, From, To, User0) -->
|
|
print_atom_argument(Atom, From, User0, OK),
|
|
( { OK = yes, From + 1 =< To } ->
|
|
print_atom_arguments(Atom, From + 1, To, User0)
|
|
;
|
|
[]
|
|
).
|
|
|
|
:- pred print_atom_argument(trace_atom::in, int::in, user_state::in, bool::out,
|
|
io__state::di, io__state::uo) is cc_multi.
|
|
|
|
print_atom_argument(Atom, ArgNum, User0, OK) -->
|
|
{ Atom = atom(_, _, Args0) },
|
|
{ maybe_filter_headvars(chosen_head_vars_presentation, Args0, Args) },
|
|
(
|
|
{ list__index1(Args, ArgNum, ArgInfo) },
|
|
{ ArgInfo = arg_info(_, _, MaybeArg) },
|
|
{ MaybeArg = yes(Arg) }
|
|
->
|
|
print(univ_value(Arg), User0 ^ outstr, decl_caller_type,
|
|
User0 ^ browser),
|
|
{ OK = yes }
|
|
;
|
|
io__write_string(User0 ^ outstr, "Invalid argument number\n"),
|
|
{ OK = no }
|
|
).
|
|
|
|
:- pred maybe_convert_dirs_to_path(maybe(list(dir)), maybe(term_path)).
|
|
:- mode maybe_convert_dirs_to_path(in, out) is det.
|
|
|
|
maybe_convert_dirs_to_path(no, no).
|
|
maybe_convert_dirs_to_path(yes(Dirs), yes(TermPath)) :-
|
|
convert_dirs_to_term_path(Dirs, TermPath).
|
|
|
|
% Reverse the first argument and append the second to it.
|
|
%
|
|
:- pred reverse_and_append(list(T), list(T), list(T)).
|
|
:- mode reverse_and_append(in, in, out) is det.
|
|
|
|
reverse_and_append([], Bs, Bs).
|
|
reverse_and_append([A | As], Bs, Cs) :-
|
|
reverse_and_append(As, [A | Bs], Cs).
|
|
|
|
%-----------------------------------------------------------------------------%
|
|
|
|
:- type user_command
|
|
---> yes % The node is correct.
|
|
; no % The node is incorrect.
|
|
; inadmissible % The node is inadmissible.
|
|
; skip % The user has no answer.
|
|
; restart % Ask the skipped questions again.
|
|
; browse_arg(int) % Browse the nth argument before
|
|
% answering.
|
|
; browse_io(int) % Browse the nth IO action before
|
|
% answering.
|
|
; print_arg(int, int) % Print the nth to the mth arguments
|
|
% before answering.
|
|
; print_io(int, int) % Print the nth to the mth IO actions
|
|
% before answering.
|
|
; pd % Commence procedural debugging from
|
|
% this point.
|
|
; abort % Abort this diagnosis session.
|
|
; help % Request help before answering.
|
|
; empty_command % User just pressed return.
|
|
; illegal_command. % None of the above.
|
|
|
|
:- pred user_help_message(user_state, io__state, io__state).
|
|
:- mode user_help_message(in, di, uo) is det.
|
|
|
|
user_help_message(User) -->
|
|
io__write_strings(User ^ outstr, [
|
|
"According to the intended interpretation of the program,",
|
|
" answer one of:\n",
|
|
"\ty\tyes\t\tthe node is correct\n",
|
|
"\tn\tno\t\tthe node is incorrect\n",
|
|
% "\ti\tinadmissible\tthe input arguments are out of range\n",
|
|
"\ts\tskip\t\tskip this question\n",
|
|
"\tr\trestart\t\task the skipped questions again\n",
|
|
"\tb <n>\tbrowse <n>\tbrowse the nth argument of the atom\n",
|
|
"\tb io <n>\tbrowse io <n>\tbrowse the atom's nth I/O action\n",
|
|
"\tp <n>\tprint <n>\tprint the nth argument of the atom\n",
|
|
"\tp <n-m>\tprint <n-m>\tprint the nth to the mth arguments of the atom\n",
|
|
"\tp io <n>\tprint io <n>\tprint the atom's nth I/O action\n",
|
|
"\tp io <n-m>\tprint io <n-m>\tprint the atom's nth to mth I/O actions\n",
|
|
"\tpd\t\tcommence procedural debugging from this point\n",
|
|
"\ta\tabort\t\t",
|
|
"abort this diagnosis session and return to mdb\n",
|
|
"\th, ?\thelp\t\tthis help message\n"
|
|
]).
|
|
|
|
:- pred user_confirm_bug_help(user_state, io__state, io__state).
|
|
:- mode user_confirm_bug_help(in, di, uo) is det.
|
|
|
|
user_confirm_bug_help(User) -->
|
|
io__write_strings(User ^ outstr, [
|
|
"Answer one of:\n",
|
|
"\ty\tyes\t\tconfirm that the suspect is a bug\n",
|
|
"\tn\tno\t\tdo not accept that the suspect is a bug\n",
|
|
% "\tb\tbrowse\t\tbrowse the suspect\n",
|
|
"\ta\tabort\t\t",
|
|
"abort this diagnosis session and return to mdb\n",
|
|
"\th, ?\thelp\t\tthis help message\n"
|
|
]).
|
|
|
|
:- pred get_command(string, user_command, user_state, user_state,
|
|
io__state, io__state).
|
|
:- mode get_command(in, out, in, out, di, uo) is det.
|
|
|
|
get_command(Prompt, Command, User, User) -->
|
|
util__trace_getline(Prompt, Result, User ^ instr, User ^ outstr),
|
|
(
|
|
{ Result = ok(String) },
|
|
{ Words = string__words(char__is_whitespace, String) },
|
|
{
|
|
Words = [CmdWord | CmdArgs],
|
|
(
|
|
cmd_handler(CmdWord, CmdHandler),
|
|
CommandPrime = CmdHandler(CmdArgs)
|
|
->
|
|
Command = CommandPrime
|
|
;
|
|
Command = illegal_command
|
|
)
|
|
;
|
|
Words = [],
|
|
Command = empty_command
|
|
}
|
|
;
|
|
{ Result = error(Error) },
|
|
{ io__error_message(Error, Msg) },
|
|
io__write_string(User ^ outstr, Msg),
|
|
io__nl(User ^ outstr),
|
|
{ Command = abort }
|
|
;
|
|
{ Result = eof },
|
|
{ Command = abort }
|
|
).
|
|
|
|
:- pred cmd_handler(string, func(list(string)) = user_command).
|
|
:- mode cmd_handler(in, out((func(in) = out is semidet))) is semidet.
|
|
|
|
cmd_handler("y", one_word_cmd(yes)).
|
|
cmd_handler("yes", one_word_cmd(yes)).
|
|
cmd_handler("n", one_word_cmd(no)).
|
|
cmd_handler("no", one_word_cmd(no)).
|
|
cmd_handler("in", one_word_cmd(inadmissible)).
|
|
cmd_handler("inadmissible", one_word_cmd(inadmissible)).
|
|
cmd_handler("s", one_word_cmd(skip)).
|
|
cmd_handler("skip", one_word_cmd(skip)).
|
|
cmd_handler("r", one_word_cmd(restart)).
|
|
cmd_handler("restart", one_word_cmd(restart)).
|
|
cmd_handler("pd", one_word_cmd(pd)).
|
|
cmd_handler("a", one_word_cmd(abort)).
|
|
cmd_handler("abort", one_word_cmd(abort)).
|
|
cmd_handler("?", one_word_cmd(help)).
|
|
cmd_handler("h", one_word_cmd(help)).
|
|
cmd_handler("help", one_word_cmd(help)).
|
|
cmd_handler("b", browse_arg_cmd).
|
|
cmd_handler("browse", browse_arg_cmd).
|
|
cmd_handler("p", print_arg_cmd).
|
|
cmd_handler("print", print_arg_cmd).
|
|
|
|
:- func one_word_cmd(user_command::in, list(string)::in) = (user_command::out)
|
|
is semidet.
|
|
|
|
one_word_cmd(Cmd, []) = Cmd.
|
|
|
|
:- func browse_arg_cmd(list(string)::in) = (user_command::out) is semidet.
|
|
|
|
browse_arg_cmd([Arg]) = browse_arg(ArgNum) :-
|
|
string__to_int(Arg, ArgNum).
|
|
browse_arg_cmd(["io", Arg]) = browse_io(ArgNum) :-
|
|
string__to_int(Arg, ArgNum).
|
|
|
|
:- func print_arg_cmd(list(string)::in) = (user_command::out) is semidet.
|
|
|
|
print_arg_cmd([Arg]) = print_arg(From, To) :-
|
|
string_to_range(Arg, From, To).
|
|
print_arg_cmd(["io", Arg]) = print_io(From, To) :-
|
|
string_to_range(Arg, From, To).
|
|
|
|
:- pred string_to_range(string::in, int::out, int::out) is semidet.
|
|
|
|
string_to_range(Arg, From, To) :-
|
|
( string__to_int(Arg, Num) ->
|
|
From = Num,
|
|
To = Num
|
|
;
|
|
[FirstStr, SecondStr] = string__words(is_dash, Arg),
|
|
string__to_int(FirstStr, First),
|
|
string__to_int(SecondStr, Second),
|
|
( First =< Second ->
|
|
From = First,
|
|
To = Second
|
|
;
|
|
From = Second,
|
|
To = First
|
|
)
|
|
).
|
|
|
|
:- pred is_dash(char::in) is semidet.
|
|
|
|
is_dash('-').
|
|
|
|
%-----------------------------------------------------------------------------%
|
|
|
|
user_confirm_bug(Bug, Response, User0, User) -->
|
|
write_decl_bug(Bug, User0),
|
|
get_command("Is this a bug? ", Command, User0, User1),
|
|
(
|
|
{ Command = yes }
|
|
->
|
|
{ Response = confirm_bug },
|
|
{ User = User1 }
|
|
;
|
|
{ Command = no }
|
|
->
|
|
{ Response = overrule_bug },
|
|
{ User = User1 }
|
|
;
|
|
{ Command = abort }
|
|
->
|
|
{ Response = abort_diagnosis },
|
|
{ User = User1 }
|
|
;
|
|
{ Command = browse_arg(ArgNum) }
|
|
->
|
|
browse_decl_bug_arg(Bug, ArgNum, User1, User2),
|
|
user_confirm_bug(Bug, Response, User2, User)
|
|
;
|
|
{ Command = browse_io(ActionNum) }
|
|
->
|
|
{ decl_bug_io_actions(Bug, IoActions) },
|
|
browse_chosen_io_action(IoActions, ActionNum, _MaybeMark,
|
|
User1, User2),
|
|
user_confirm_bug(Bug, Response, User2, User)
|
|
;
|
|
user_confirm_bug_help(User1),
|
|
user_confirm_bug(Bug, Response, User1, User)
|
|
).
|
|
|
|
%-----------------------------------------------------------------------------%
|
|
|
|
% Returns the caller type we want to use throughout the
|
|
% declarative debugger.
|
|
:- func decl_caller_type = browse_caller_type.
|
|
|
|
decl_caller_type = print.
|
|
|
|
% Display the node in user readable form on the current
|
|
% output stream.
|
|
%
|
|
:- pred write_decl_question(decl_question(T)::in, user_state::in,
|
|
io__state::di, io__state::uo) is cc_multi.
|
|
|
|
write_decl_question(wrong_answer(_, Atom), User) -->
|
|
write_decl_final_atom(User, "", decl_caller_type, Atom).
|
|
|
|
write_decl_question(missing_answer(_, Call, Solns), User) -->
|
|
write_decl_init_atom(User, "Call ", decl_caller_type, Call),
|
|
(
|
|
{ Solns = [] }
|
|
->
|
|
io__write_string(User ^ outstr, "No solutions.\n")
|
|
;
|
|
io__write_string(User ^ outstr, "Solutions:\n"),
|
|
list__foldl(write_decl_final_atom(User, "\t", print_all), Solns)
|
|
).
|
|
|
|
write_decl_question(unexpected_exception(_, Call, Exception), User) -->
|
|
write_decl_init_atom(User, "Call ", decl_caller_type, Call),
|
|
io__write_string(User ^ outstr, "Throws "),
|
|
io__write(User ^ outstr, include_details_cc, univ_value(Exception)),
|
|
io__nl(User ^ outstr).
|
|
|
|
:- pred write_decl_bug(decl_bug::in, user_state::in,
|
|
io__state::di, io__state::uo) is cc_multi.
|
|
|
|
write_decl_bug(e_bug(EBug), User) -->
|
|
(
|
|
{ EBug = incorrect_contour(Atom, _, _) },
|
|
io__write_string(User ^ outstr, "Found incorrect contour:\n"),
|
|
write_decl_final_atom(User, "", decl_caller_type, Atom)
|
|
;
|
|
{ EBug = partially_uncovered_atom(Atom, _) },
|
|
io__write_string(User ^ outstr,
|
|
"Found partially uncovered atom:\n"),
|
|
write_decl_init_atom(User, "", decl_caller_type, Atom)
|
|
;
|
|
{ EBug = unhandled_exception(Atom, Exception, _) },
|
|
io__write_string(User ^ outstr, "Found unhandled exception:\n"),
|
|
write_decl_init_atom(User, "", decl_caller_type, Atom),
|
|
io__write(User ^ outstr, include_details_cc,
|
|
univ_value(Exception)),
|
|
io__nl(User ^ outstr)
|
|
).
|
|
|
|
write_decl_bug(i_bug(IBug), User) -->
|
|
{ IBug = inadmissible_call(Parent, _, Call, _) },
|
|
io__write_string(User ^ outstr, "Found inadmissible call:\n"),
|
|
write_decl_atom(User, "Parent ", decl_caller_type, init(Parent)),
|
|
write_decl_atom(User, "Call ", decl_caller_type, init(Call)).
|
|
|
|
:- pred write_decl_init_atom(user_state::in, string::in, browse_caller_type::in,
|
|
init_decl_atom::in, io__state::di, io__state::uo) is cc_multi.
|
|
|
|
write_decl_init_atom(User, Indent, CallerType, InitAtom) -->
|
|
write_decl_atom(User, Indent, CallerType, init(InitAtom)).
|
|
|
|
:- pred write_decl_final_atom(user_state::in, string::in,
|
|
browse_caller_type::in, final_decl_atom::in, io__state::di,
|
|
io__state::uo) is cc_multi.
|
|
|
|
write_decl_final_atom(User, Indent, CallerType, FinalAtom) -->
|
|
write_decl_atom(User, Indent, CallerType, final(FinalAtom)).
|
|
|
|
:- pred write_decl_atom(user_state::in, string::in, browse_caller_type::in,
|
|
some_decl_atom::in, io__state::di, io__state::uo) is cc_multi.
|
|
|
|
write_decl_atom(User, Indent, CallerType, DeclAtom) -->
|
|
io__write_string(User ^ outstr, Indent),
|
|
{ unravel_decl_atom(DeclAtom, TraceAtom, IoActions) },
|
|
{ TraceAtom = atom(PredOrFunc, Functor, Args0) },
|
|
{ Which = chosen_head_vars_presentation },
|
|
{ maybe_filter_headvars(Which, Args0, Args1) },
|
|
{ list__map(trace_atom_arg_to_univ, Args1, Args) },
|
|
%
|
|
% Call the term browser to print the atom (or part of it
|
|
% up to a size limit) as a goal.
|
|
%
|
|
browse__print_synthetic(Functor, Args, is_function(PredOrFunc),
|
|
User ^ outstr, CallerType, User ^ browser),
|
|
write_io_actions(User, IoActions).
|
|
|
|
:- pred trace_atom_arg_to_univ(trace_atom_arg::in, univ::out) is det.
|
|
|
|
trace_atom_arg_to_univ(TraceAtomArg, Univ) :-
|
|
MaybeUniv = TraceAtomArg ^ arg_value,
|
|
(
|
|
MaybeUniv = yes(Univ)
|
|
;
|
|
MaybeUniv = no,
|
|
Univ = univ('_' `with_type` unbound)
|
|
).
|
|
|
|
:- pred write_io_actions(user_state::in, list(io_action)::in, io__state::di,
|
|
io__state::uo) is cc_multi.
|
|
|
|
write_io_actions(User, IoActions) -->
|
|
{ list__length(IoActions, NumIoActions) },
|
|
( { NumIoActions = 0 } ->
|
|
[]
|
|
;
|
|
( { NumIoActions = 1 } ->
|
|
io__write_string(User ^ outstr, "1 io action:")
|
|
;
|
|
io__write_int(User ^ outstr, NumIoActions),
|
|
io__write_string(User ^ outstr, " io actions:")
|
|
),
|
|
{ NumPrinted = get_num_printed_io_actions(User ^ browser) },
|
|
( { NumIoActions =< NumPrinted } ->
|
|
io__nl(User ^ outstr),
|
|
list__foldl(print_io_action(User), IoActions)
|
|
;
|
|
io__write_string(User ^ outstr, " too many to show"),
|
|
io__nl(User ^ outstr)
|
|
)
|
|
).
|
|
|
|
:- pred print_io_action(user_state::in, io_action::in,
|
|
io__state::di, io__state::uo) is cc_multi.
|
|
|
|
print_io_action(User, IoAction) -->
|
|
{ io_action_to_synthetic_term(IoAction, ProcName, Args, IsFunc) },
|
|
browse__print_synthetic(ProcName, Args, IsFunc, User ^ outstr,
|
|
print_all, User ^ browser).
|
|
|
|
%-----------------------------------------------------------------------------%
|