From e3cb9e8fbd8eda9581adcb092f99df088b53a27f Mon Sep 17 00:00:00 2001 From: ClemensBuechner Date: Mon, 8 Jan 2024 16:38:40 +0100 Subject: [PATCH 1/2] Prevent returning nullpointer if data array is empty. --- src/search/lp/cplex_solver_interface.h | 40 ++++++++++++++++++-------- 1 file changed, 28 insertions(+), 12 deletions(-) diff --git a/src/search/lp/cplex_solver_interface.h b/src/search/lp/cplex_solver_interface.h index 5ba35cd532..5737d63de1 100644 --- a/src/search/lp/cplex_solver_interface.h +++ b/src/search/lp/cplex_solver_interface.h @@ -11,6 +11,22 @@ #include namespace lp { + +template +static T* to_cplex_array(std::vector &v) { + /* + CPLEX expects a non-nullptr even for empty arrays but the C++ standard + does not guarantee any particular value for data for empty vectors (see + issue1111). + */ + if (v.empty()) { + static T dummy; + return &dummy; + } else { + return v.data(); + } +} + class CplexSolverInterface : public SolverInterface { CPXENVptr env; CPXLPptr problem; @@ -90,10 +106,10 @@ class CplexSolverInterface : public SolverInterface { void assign_row_by_row( const named_vector::NamedVector &constraints); - double *get_coefficients() {return coefficients.data();} - int *get_indices() {return indices.data();} - int *get_starts() {return starts.data();} - int *get_counts() {return counts.data();} + double *get_coefficients() {return to_cplex_array(coefficients);} + int *get_indices() {return to_cplex_array(indices);} + int *get_starts() {return to_cplex_array(starts);} + int *get_counts() {return to_cplex_array(counts);} int get_num_nonzeros() {return coefficients.size();} }; @@ -108,10 +124,10 @@ class CplexSolverInterface : public SolverInterface { std::vector objective; public: void assign(const named_vector::NamedVector &variables); - double *get_lb() {return lb.data();} - double *get_ub() {return ub.data();} - char *get_type() {return type.data();} - double *get_objective() {return objective.data();} + double *get_lb() {return to_cplex_array(lb);} + double *get_ub() {return to_cplex_array(ub);} + char *get_type() {return to_cplex_array(type);} + double *get_objective() {return to_cplex_array(objective);} }; class CplexRowsInfo { @@ -131,10 +147,10 @@ class CplexSolverInterface : public SolverInterface { std::vector range_indices; public: void assign(const named_vector::NamedVector &constraints, int offset = 0, bool dense_range_values = true); - double *get_rhs() {return rhs.data();} - char *get_sense() {return sense.data();} - double *get_range_values() {return range_values.data();} - int *get_range_indices() {return range_indices.data();} + double *get_rhs() {return to_cplex_array(rhs);} + char *get_sense() {return to_cplex_array(sense);} + double *get_range_values() {return to_cplex_array(range_values);} + int *get_range_indices() {return to_cplex_array(range_indices);} int get_num_ranged_rows() {return range_indices.size();} }; From fd9345bf6243c4c32b46bd6b590f17745b479055 Mon Sep 17 00:00:00 2001 From: ClemensBuechner Date: Tue, 9 Jan 2024 11:17:04 +0100 Subject: [PATCH 2/2] Fix style. --- src/search/lp/cplex_solver_interface.h | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/src/search/lp/cplex_solver_interface.h b/src/search/lp/cplex_solver_interface.h index 5737d63de1..8104a23eed 100644 --- a/src/search/lp/cplex_solver_interface.h +++ b/src/search/lp/cplex_solver_interface.h @@ -11,9 +11,8 @@ #include namespace lp { - template -static T* to_cplex_array(std::vector &v) { +static T *to_cplex_array(std::vector &v) { /* CPLEX expects a non-nullptr even for empty arrays but the C++ standard does not guarantee any particular value for data for empty vectors (see