From b1ea628b0282441cb5231ac00987d43c19f6929b Mon Sep 17 00:00:00 2001 From: Ihnatus Date: Tue, 13 Dec 2022 00:12:36 +0300 Subject: [PATCH] Evaluate requirements in a vector with an arbitrary callback Provides the evaluator tri_reqs_cb_active() that takes the requirement vector, the callback, the context and some side data and tells fc_tristate for the vector, in process it may collect the uncertain requirements in another vector. Also, provides an example callback default_tester_cb() that consults a table of requirements that are considered true disregarding the context, and if neither of those requirements gives a definite value for given one from the vector, evaluates it usual way. All this is a base for upcoming patches. See OSDN#46266. Signed-off-by: Ihnatus --- common/requirements.c | 112 ++++++++++++++++++++++++++++++++++++++++++ common/requirements.h | 27 ++++++++++ 2 files changed, 139 insertions(+) diff --git a/common/requirements.c b/common/requirements.c index 1ae5474d6f..5b2e91ef5f 100644 --- a/common/requirements.c +++ b/common/requirements.c @@ -1522,6 +1522,19 @@ static bool present_implies_not_present(const struct requirement *req1, &present->source); } +/**********************************************************************//** + Returns TRUE if req2 is always fulfilled when req1 is (i.e. req1 => req2) +**************************************************************************/ +bool req_implies_req(const struct requirement *req1, + const struct requirement *req2) +{ + struct requirement nreq2; + + req_copy(&nreq2, req2); + nreq2.present = !nreq2.present; + return are_requirements_contradictions(req1, &nreq2); +} + /**********************************************************************//** Returns TRUE if req1 and req2 contradicts each other. @@ -4905,6 +4918,32 @@ enum fc_tristate tri_req_present(const struct req_context *context, return req_definitions[req->source.kind].cb(context, other_player, req); } +/**********************************************************************//** + Evaluates req in context with other_player to fc_tristate. + + context may be NULL. This is equivalent to passing an empty context. + + Fields of context that are NULL are considered unspecified + and will produce TRI_MAYBE if req needs them to evaluate. +**************************************************************************/ +enum fc_tristate tri_req_active(const struct req_context *context, + const struct player *other_player, + const struct requirement *req) +{ + enum fc_tristate eval = tri_req_present(context, other_player, req); + + if (!req->present) { + if (TRI_NO == eval) { + return TRI_YES; + } + if (TRI_YES == eval) { + return TRI_NO; + } + } + + return eval; +} + /**********************************************************************//** Checks the requirement(s) to see if they are active on the given target. @@ -4955,6 +4994,79 @@ bool are_reqs_active_ranges(const enum req_range min_range, return TRUE; } +/**********************************************************************//** + A tester callback for tri_reqs_cb_active() that uses the default + requirement calculation except for requirements in data[n_data] array + and ones implied by them or contradicting them +**************************************************************************/ +enum fc_tristate default_tester_cb + (const struct req_context *context, + const struct player *other_player, + const struct requirement *req, + void *data, int n_data) +{ + fc_assert_ret_val(data || n_data == 0, TRI_NO); + + for (int i = 0; i < n_data; i++) { + if (are_requirements_contradictions(&((struct requirement *) data)[i], + req)) { + return TRI_NO; + } else if (req_implies_req(&((struct requirement *) data)[i], req)) { + return TRI_YES; + } + } + return tri_req_active(context, other_player, req); +} + +/**********************************************************************//** + Test requirements in reqs with tester according to (data, n_data) + and give the resulting tristate. + If maybe_reqs is not NULL, copies requirements that are evaluated + to TRI_MAYBE into it (stops as soon as one evaluates to TRI_NO). +**************************************************************************/ +enum fc_tristate + tri_reqs_cb_active(const struct req_context *context, + const struct player *other_player, + const struct requirement_vector *reqs, + struct requirement_vector *maybe_reqs, + req_tester_cb tester, + void *data, int n_data) +{ + bool active = TRUE; + bool certain = TRUE; + int sz = requirement_vector_size(reqs); + + fc_assert_ret_val(NULL != tester, TRI_NO); + requirement_vector_iterate(reqs, preq) { + switch(tester(context, other_player, preq, + data, n_data)) { + case TRI_NO: + active = FALSE; + certain = TRUE; + break; + case TRI_YES: + break; + case TRI_MAYBE: + certain = FALSE; + if (maybe_reqs) { + requirement_vector_append(maybe_reqs, *preq); + } + break; + default: + fc_assert(FALSE); + active = FALSE; + } + if (!active) { + break; + } + sz--; + } requirement_vector_iterate_end; + + return certain ? (active ? TRI_YES : TRI_NO) : TRI_MAYBE; +} + + + /**********************************************************************//** Gives a suggestion may req ever evaluate to another value with given context. (The other player is not supplied since it has no value diff --git a/common/requirements.h b/common/requirements.h index 1eeccb3ba9..fa7e7881e2 100644 --- a/common/requirements.h +++ b/common/requirements.h @@ -153,9 +153,15 @@ bool are_requirements_equal(const struct requirement *req1, bool are_requirements_contradictions(const struct requirement *req1, const struct requirement *req2); +bool req_implies_req(const struct requirement *req1, + const struct requirement *req2); + bool does_req_contradicts_reqs(const struct requirement *req, const struct requirement_vector *vec); +enum fc_tristate tri_req_active(const struct req_context *context, + const struct player *other_player, + const struct requirement *req); bool is_req_active(const struct req_context *context, const struct player *other_player, const struct requirement *req, @@ -171,6 +177,27 @@ bool are_reqs_active_ranges(const enum req_range min_range, const struct requirement_vector *reqs, const enum req_problem_type prob_type); +/* Type of a callback that tests requirements due to a context + * and something else in some manner different from tri_req_active() */ +typedef enum fc_tristate + (*req_tester_cb)(const struct req_context *context, + const struct player *other_player, + const struct requirement *req, + void *data, int n_data); + +enum fc_tristate +default_tester_cb(const struct req_context *context, + const struct player *other_player, + const struct requirement *req, + void *data, int n_data); +enum fc_tristate + tri_reqs_cb_active(const struct req_context *context, + const struct player *other_player, + const struct requirement_vector *reqs, + struct requirement_vector *maybe_reqs, + req_tester_cb tester, + void *data, int n_data); + enum req_unchanging_status is_req_unchanging(const struct req_context *context, const struct requirement *req); -- 2.37.2