mirror of
https://github.com/Mercury-Language/mercury.git
synced 2025-12-08 18:34:00 +00:00
Discussion of these changes can be found on the Mercury developers
mailing list archives from June 2018.
COPYING.LIB:
Add a special linking exception to the LGPL.
*:
Update references to COPYING.LIB.
Clean up some minor errors that have accumulated in copyright
messages.
266 lines
11 KiB
C
266 lines
11 KiB
C
// vim: ts=4 sw=4 expandtab ft=c
|
|
|
|
// Copyright (C) 2003-2006 The University of Melbourne.
|
|
// Copyright (C) 2016, 2018 The Mercury team.
|
|
// This file is distributed under the terms specified in COPYING.LIB.
|
|
|
|
// mercury_minimal_model.h - definitions of some basic stuff used for the stack
|
|
// copy style of minimal model tabling. For tabling code, the Mercury compiler
|
|
// (compiler/table_gen.m) generates references to special procedures defined
|
|
// in library/table_builtin.m. The types and macros defined here are used
|
|
// by the procedures defined in library/table_builtin.m.
|
|
|
|
#ifndef MERCURY_MINIMAL_MODEL_H
|
|
#define MERCURY_MINIMAL_MODEL_H
|
|
|
|
#include "mercury_imp.h"
|
|
#include "mercury_tabling.h"
|
|
#include "mercury_reg_workarounds.h"
|
|
#include "mercury_goto.h" // for MR_declare_entry
|
|
#include <stdio.h>
|
|
|
|
#ifdef MR_USE_MINIMAL_MODEL_STACK_COPY
|
|
|
|
// The saved state of a generator or a consumer. While consumers get
|
|
// suspended while they are waiting for generators to produce more solutions,
|
|
// generators need their state saved when they restore the state of a consumer
|
|
// to consume a new solution.
|
|
//
|
|
// The saved state contains copies of
|
|
//
|
|
// - several virtual machine registers:
|
|
// MR_succip, MR_sp, MR_curfr and MR_maxfr
|
|
//
|
|
// - segments of the nondet and det stacks:
|
|
// the parts that cannot possibly change between the times of saving
|
|
// and restoring the saved state are not saved.
|
|
//
|
|
// The segments are described by three fields each. The *_real_start
|
|
// field gives the address of the first word in the real stack
|
|
// that is part of the saved segment, the *_block_size field
|
|
// gives the size of the saved segment in words, and the *_saved_block
|
|
// field points to the area of memory containing the saved segment.
|
|
//
|
|
// - the entire generator stack and the entire cut stack:
|
|
// they are usually so small, it is faster to save them all
|
|
// than to figure out which parts need saving.
|
|
//
|
|
// Each stack is described by its size in words and a pointer to
|
|
// an area of memory containing the entire saved stack.
|
|
|
|
typedef struct {
|
|
MR_Code *MR_ss_succ_ip;
|
|
MR_Word *MR_ss_s_p;
|
|
MR_Word *MR_ss_cur_fr;
|
|
MR_Word *MR_ss_max_fr;
|
|
MR_Word *MR_ss_common_ancestor_fr;
|
|
MR_Integer MR_ss_gen_sp;
|
|
|
|
MR_Word *MR_ss_non_stack_real_start;
|
|
MR_Integer MR_ss_non_stack_block_size;
|
|
MR_Word *MR_ss_non_stack_saved_block;
|
|
|
|
MR_Word *MR_ss_det_stack_real_start;
|
|
MR_Integer MR_ss_det_stack_block_size;
|
|
MR_Word *MR_ss_det_stack_saved_block;
|
|
|
|
MR_Integer MR_ss_gen_stack_real_start;
|
|
MR_Integer MR_ss_gen_stack_block_size;
|
|
MR_GenStackFrame *MR_ss_gen_stack_saved_block;
|
|
|
|
MR_Integer MR_ss_cut_next;
|
|
MR_CutStackFrame *MR_ss_cut_stack_saved_block;
|
|
|
|
MR_Integer MR_ss_pneg_next;
|
|
MR_PNegStackFrame *MR_ss_pneg_stack_saved_block;
|
|
|
|
#ifdef MR_MINIMAL_MODEL_DEBUG
|
|
// To make it possible to generate more detailed debugging output.
|
|
const MR_LabelLayout *MR_ss_top_layout;
|
|
#endif
|
|
} MR_SavedState;
|
|
|
|
// The state of a consumer subgoal.
|
|
//
|
|
// The subgoal field points to the generator subgoal. If it is NULL, then the
|
|
// consumer has logically been deleted.
|
|
//
|
|
// The num_returned answers field gives the number of answers returned to this
|
|
// consumer so far.
|
|
//
|
|
// The answer list fields points into the answer list of the generator
|
|
// subgoal. The list of answers it points to expands transparently whenever
|
|
// the generator subgoal adds an answer to its answer list.
|
|
|
|
struct MR_Consumer_Struct {
|
|
MR_SavedState MR_cns_saved_state;
|
|
MR_Subgoal *MR_cns_subgoal;
|
|
MR_Integer MR_cns_num_returned_answers;
|
|
MR_AnswerList *MR_cns_remaining_answer_list_ptr;
|
|
};
|
|
|
|
struct MR_ConsumerListNode_Struct {
|
|
MR_Consumer *MR_cl_item;
|
|
MR_ConsumerList MR_cl_next;
|
|
};
|
|
|
|
// The following structure is used to hold the state and variables used in
|
|
// the table_mm_completion procedure.
|
|
|
|
typedef struct {
|
|
MR_SavedState MR_ri_leader_state;
|
|
MR_SubgoalList MR_ri_subgoal_list;
|
|
MR_Subgoal *MR_ri_cur_subgoal;
|
|
MR_ConsumerList MR_ri_consumer_list; // for the cur subgoal
|
|
MR_Consumer *MR_ri_cur_consumer;
|
|
MR_Code *MR_ri_saved_succip;
|
|
} MR_CompletionInfo;
|
|
|
|
struct MR_SubgoalListNode_Struct {
|
|
MR_Subgoal *MR_sl_item;
|
|
MR_SubgoalList MR_sl_next;
|
|
};
|
|
|
|
// The subgoal structure represents a subgoal, i.e. a call to a minimal model
|
|
// predicate with a given set of input arguments.
|
|
//
|
|
// The MR_sg_status obviously gives the current status of the subgoal.
|
|
//
|
|
// The MR_sg_leader field points to the leader of the clique of subgoals this
|
|
// subgoal is a member of, if the leader is not the subgoal itself. If the
|
|
// subgoal is the leader of its clique, then this field will be NULL.
|
|
//
|
|
// The MR_sg_followers field lists all the subgoals that follow this one.
|
|
// Each subgoal occurs in its own followers list.
|
|
//
|
|
// The MR_sg_followers_tail field points to the NULL pointer at the tail of
|
|
// the MR_sg_followers list, to allow us to add new elements at the tail in
|
|
// constant time.
|
|
//
|
|
// The MR_sg_completion_info field, when non-NULL, points to a data structure
|
|
// containing the local variables of the algorithm executed by the builtin
|
|
// predicate table_mm_completion. The execution of that algorithm invokes
|
|
// general Mercury code and moves stack segments around many times, which
|
|
// is why its local variables cannot be stored in a stack frame.
|
|
//
|
|
// The MR_sg_answer_table field points to the trie of answers returned for
|
|
// this subgoal so far (to enable efficient checks for duplicate answers).
|
|
//
|
|
// The MR_sg_answer_list field also contains the answers so far, in a list.
|
|
// The answers will be returned to consumers in the order given by this list.
|
|
// The MR_sg_answer_list_tail field points to the NULL pointer at the tail
|
|
// of this list, to allow us to add new elements to the list at the tail in
|
|
// constant time.
|
|
//
|
|
// The MR_sg_num_ans field gives the number of answers computed so far, i.e.
|
|
// the number of answers in MR_sg_answer_table and MR_sg_answer_list.
|
|
//
|
|
// The MR_sg_consumer_list gives the list of consumer goals, with the field
|
|
// MR_sg_consumer_list_tail allowing fast appending to the end.
|
|
//
|
|
// The MR_sg_generator_fr points to the generator's nondet stack frame.
|
|
//
|
|
// The MR_sg_deepest_nca_fr points to the deepest nondet stack frame that is
|
|
// a nearest common ancestor (nca) of the generator and one of its consumers.
|
|
|
|
struct MR_Subgoal_Struct {
|
|
MR_TrieNode MR_sg_back_ptr;
|
|
MR_SubgoalStatus MR_sg_status;
|
|
MR_Subgoal *MR_sg_leader;
|
|
MR_SubgoalList MR_sg_followers;
|
|
MR_SubgoalList *MR_sg_followers_tail;
|
|
MR_CompletionInfo *MR_sg_completion_info;
|
|
MR_TableNode MR_sg_answer_table;
|
|
MR_Integer MR_sg_num_ans;
|
|
MR_AnswerList MR_sg_answer_list;
|
|
MR_AnswerList *MR_sg_answer_list_tail;
|
|
MR_ConsumerList MR_sg_consumer_list;
|
|
MR_ConsumerList *MR_sg_consumer_list_tail;
|
|
MR_Word *MR_sg_generator_fr;
|
|
MR_Word *MR_sg_deepest_nca_fr;
|
|
#ifdef MR_MINIMAL_MODEL_DEBUG
|
|
const MR_ProcLayout *MR_sg_proc_layout;
|
|
#endif
|
|
};
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
extern const MR_ProcLayout *MR_subgoal_debug_cur_proc;
|
|
|
|
extern void MR_enter_consumer_debug(MR_Consumer *consumer);
|
|
extern MR_ConsumerDebug *MR_lookup_consumer_debug_addr(MR_Consumer *consumer);
|
|
extern MR_ConsumerDebug *MR_lookup_consumer_debug_num(int consumer_index);
|
|
extern const char *MR_consumer_debug_name(MR_ConsumerDebug *consumer_dbg);
|
|
extern const char *MR_consumer_addr_name(MR_Consumer *consumer);
|
|
extern const char *MR_consumer_num_name(int consumer_index);
|
|
|
|
extern void MR_enter_subgoal_debug(MR_Subgoal *subgoal);
|
|
extern MR_SubgoalDebug *MR_lookup_subgoal_debug_addr(MR_Subgoal *subgoal);
|
|
extern MR_SubgoalDebug *MR_lookup_subgoal_debug_num(int subgoal_index);
|
|
extern const char *MR_subgoal_debug_name(MR_SubgoalDebug *subgoal_debug);
|
|
extern const char *MR_subgoal_addr_name(MR_Subgoal *subgoal);
|
|
extern const char *MR_subgoal_num_name(int subgoal_index);
|
|
extern const char *MR_subgoal_status(MR_SubgoalStatus status);
|
|
|
|
extern void MR_print_subgoal_debug(FILE *fp,
|
|
const MR_ProcLayout *proc,
|
|
MR_SubgoalDebug *subgoal_debug);
|
|
extern void MR_print_subgoal(FILE *fp, const MR_ProcLayout *proc,
|
|
MR_Subgoal *subgoal);
|
|
extern void MR_print_consumer_debug(FILE *fp,
|
|
const MR_ProcLayout *proc,
|
|
MR_ConsumerDebug *consumer_debug);
|
|
extern void MR_print_consumer(FILE *fp, const MR_ProcLayout *proc,
|
|
MR_Consumer *consumer);
|
|
|
|
extern MR_Subgoal *MR_setup_subgoal(MR_TrieNode);
|
|
|
|
#ifdef MR_TABLE_STATISTICS
|
|
extern void MR_minimal_model_report_stats(FILE *fp);
|
|
extern int MR_minmodel_stats_cnt_dupl_check;
|
|
extern int MR_minmodel_stats_cnt_dupl_check_not_dupl;
|
|
// XXX
|
|
#endif
|
|
|
|
#endif // MR_USE_MINIMAL_MODEL_STACK_COPY
|
|
|
|
#ifdef MR_HIGHLEVEL_CODE
|
|
|
|
extern void MR_CALL
|
|
mercury__table_builtin__table_mm_completion_1_p_0(
|
|
MR_C_Pointer subgoal_table_node, MR_C_Pointer *answer_block,
|
|
MR_Cont cont, void *cont_env_ptr);
|
|
extern void MR_CALL
|
|
mercury__table_builtin__table_mm_suspend_consumer_2_p_0(
|
|
MR_C_Pointer subgoal_table_node);
|
|
extern void MR_CALL
|
|
mercury__table_builtin__table_mm_return_all_nondet_2_p_0(
|
|
MR_C_Pointer answer_list, MR_C_Pointer answer_block);
|
|
extern void MR_CALL
|
|
mercury__table_builtin__table_mm_return_all_multi_2_p_0(
|
|
MR_C_Pointer answer_list, MR_C_Pointer answer_block);
|
|
|
|
#else // ! MR_HIGHLEVEL_CODE
|
|
|
|
#define MR_MMSC_SUSPEND_ENTRY \
|
|
MR_proc_entry_user_name(table_builtin, \
|
|
table_mm_suspend_consumer, 2, 0)
|
|
#define MR_MMSC_COMPLETION_ENTRY \
|
|
MR_proc_entry_user_name(table_builtin, \
|
|
table_mm_completion, 1, 0)
|
|
#define MR_MMSC_RET_ALL_NONDET_ENTRY \
|
|
MR_proc_entry_user_name(table_builtin, \
|
|
table_mm_return_all_nondet, 2, 0)
|
|
#define MR_MMSC_RET_ALL_MULTI_ENTRY \
|
|
MR_proc_entry_user_name(table_builtin, \
|
|
table_mm_return_all_multi, 2, 0)
|
|
|
|
MR_declare_entry(MR_MMSC_SUSPEND_ENTRY);
|
|
MR_declare_entry(MR_MMSC_COMPLETION_ENTRY);
|
|
MR_declare_entry(MR_MMSC_RET_ALL_NONDET_ENTRY);
|
|
MR_declare_entry(MR_MMSC_RET_ALL_MULTI_ENTRY);
|
|
|
|
#endif // !MR_HIGHLEVEL_CODE
|
|
|
|
#endif // MERCURY_MINIMAL_MODEL_H
|