Scippy

SCIP

Solving Constraint Integer Programs

cons_pseudoboolean.c
Go to the documentation of this file.
1 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2 /* */
3 /* This file is part of the program and library */
4 /* SCIP --- Solving Constraint Integer Programs */
5 /* */
6 /* Copyright (c) 2002-2023 Zuse Institute Berlin (ZIB) */
7 /* */
8 /* Licensed under the Apache License, Version 2.0 (the "License"); */
9 /* you may not use this file except in compliance with the License. */
10 /* You may obtain a copy of the License at */
11 /* */
12 /* http://www.apache.org/licenses/LICENSE-2.0 */
13 /* */
14 /* Unless required by applicable law or agreed to in writing, software */
15 /* distributed under the License is distributed on an "AS IS" BASIS, */
16 /* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. */
17 /* See the License for the specific language governing permissions and */
18 /* limitations under the License. */
19 /* */
20 /* You should have received a copy of the Apache-2.0 license */
21 /* along with SCIP; see the file LICENSE. If not visit scipopt.org. */
22 /* */
23 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
24 
25 /**@file cons_pseudoboolean.c
26  * @ingroup DEFPLUGINS_CONS
27  * @brief constraint handler for pseudo Boolean constraints
28  * @author Gerald Gamrath
29  * @author Stefan Heinz
30  * @author Michael Winkler
31  *
32  *
33  * The constraint handler deals with pseudo Boolean constraints. These are constraints of the form
34  * \f[
35  * \mbox{lhs} \leq \sum_{k=0}^m c_k \cdot x_k + \sum_{i=0}^n c_i \cdot \prod_{j \in I_i} x_j \leq \mbox{rhs}
36  * \f]
37  * where all x are binary and all c are integer
38  *
39  * @todo Add eventhandling.
40  */
41 
42 /*---+----1----+----2----+----3----+----4----+----5----+----6----+----7----+----8----+----9----+----0----+----1----+----2*/
43 
44 #include "blockmemshell/memory.h"
45 #include "scip/cons_and.h"
46 #include "scip/cons_indicator.h"
47 #include "scip/cons_knapsack.h"
48 #include "scip/cons_linear.h"
49 #include "scip/cons_logicor.h"
51 #include "scip/cons_setppc.h"
52 #include "scip/cons_xor.h"
53 #include "scip/debug.h"
54 #include "scip/pub_cons.h"
55 #include "scip/pub_message.h"
56 #include "scip/pub_misc.h"
57 #include "scip/pub_misc_sort.h"
58 #include "scip/pub_var.h"
59 #include "scip/scip_cons.h"
60 #include "scip/scip_copy.h"
61 #include "scip/scip_general.h"
62 #include "scip/scip_mem.h"
63 #include "scip/scip_message.h"
64 #include "scip/scip_numerics.h"
65 #include "scip/scip_param.h"
66 #include "scip/scip_prob.h"
67 #include "scip/scip_sol.h"
68 #include "scip/scip_var.h"
69 #include <string.h>
70 
71 #ifdef WITHEQKNAPSACK
72 #include "scip/cons_eqknapsack.h"
73 #endif
74 
75 /* constraint handler properties */
76 #define CONSHDLR_NAME "pseudoboolean"
77 #define CONSHDLR_DESC "constraint handler dealing with pseudo Boolean constraints"
78 #define CONSHDLR_ENFOPRIORITY -1000000 /**< priority of the constraint handler for constraint enforcing */
79 #define CONSHDLR_CHECKPRIORITY -5000000 /**< priority of the constraint handler for checking feasibility */
80 #define CONSHDLR_EAGERFREQ 100 /**< frequency for using all instead of only the useful constraints in separation,
81  * propagation and enforcement, -1 for no eager evaluations, 0 for first only */
82 #define CONSHDLR_MAXPREROUNDS -1 /**< maximal number of presolving rounds the constraint handler participates in (-1: no limit) */
83 #define CONSHDLR_NEEDSCONS TRUE /**< should the constraint handler be skipped, if no constraints are available? */
84 
85 #define CONSHDLR_PRESOLTIMING SCIP_PRESOLTIMING_MEDIUM /**< presolving timing of the constraint handler (fast, medium, or exhaustive) */
86 
87 #define DEFAULT_DECOMPOSENORMALPBCONS FALSE /**< decompose all normal pseudo boolean constraint into a "linear" constraint and "and" constraints */
88 #define DEFAULT_DECOMPOSEINDICATORPBCONS TRUE /**< decompose all indicator pseudo boolean constraint into a "linear" constraint and "and" constraints */
89 
90 #define DEFAULT_SEPARATENONLINEAR TRUE /**< if decomposed, should the nonlinear constraints be separated during LP processing */
91 #define DEFAULT_PROPAGATENONLINEAR TRUE /**< if decomposed, should the nonlinear constraints be propagated during node processing */
92 #define DEFAULT_REMOVABLENONLINEAR TRUE /**< if decomposed, should the nonlinear constraints be removable */
93 #define USEINDICATOR TRUE
94 #define NONLINCONSUPGD_PRIORITY 60000 /**< priority of upgrading nonlinear constraints */
95 
96 /* remove this line to compile the upgrade from nonlinear to pseudoboolean constraints */
97 #undef NONLINCONSUPGD_PRIORITY /*lint !e750*/
98 
99 /*
100  * Data structures
101  */
102 #define HASHSIZE_PSEUDOBOOLEANNONLINEARTERMS 500 /**< minimal size of hash table in and constraint tables */
104 
105 /* - create special linear(knapsack, setppc, logicor, (eqknapsack)) and and-constraints with check flags FALSE, to
106  * get smaller amount of locks on the term variables, do all presolving ...?! in these constraint handlers
107  *
108  * - do the checking here, lock and-resultants in both directions and all and-variables according to their
109  * coefficients and sides of the constraint,
110  * @note this only works if the and-resultant has no objective cofficient, otherwise we need to lock variables also in both directions
111  *
112  * - need to keep and constraint pointer for special propagations like if two ands are due to their variables in
113  * one clique, add this cliques of and-resultants
114  *
115  * - do special presolving like on instance :
116  * check/IP/PseudoBoolean/normalized-PB07/OPT-SMALLINT-NLC/submittedPB07/manquinho/bsg/normalized-bsg_1000_25_1.opb.gz
117  *
118  * there exist constraint like: 1 x1 x2 + 1 x1 x3 + 1 x1 x4 + 1 x1 x5 <= 1 ;
119  * which "equals" a linear constraint: 3 x1 + x2 + x3 + x4 + x5 <= 4 ;
120  *
121  * in more general terms: 1 x1 x2 x3 x4 + 1 x1 x2 x5 x6 x7 + 1 x1 x2 x8 x9 <= 1 ;
122  * which "equals" a pseudoboolean constraint: 2 x1 + 2 x2 + 1 x3 x4 + 1 x5 x6 x7 + 1 x8 x9 <= 5 ;
123  *
124  * in an even more general terms: 5 x1 x2 x3 x4 + 1 x1 x2 x5 x6 x7 + 1 x1 x2 x8 x9 <= 6 ;
125  * equals(should the knapsack do) 1 x1 x2 x3 x4 + 1 x1 x2 x5 x6 x7 + 1 x1 x2 x8 x9 <= 2 ;
126  * which "equals" a pseudoboolean constraint: 2 x1 + 2 x2 + 1 x3 x4 + 1 x5 x6 x7 + 1 x8 x9 <= 6 ;
127  * ( without knapsack 7 x1 + 7 x2 + 5 x3 x4 + 1 x5 x6 x7 + 1 x8 x9 <= 20 ; )
128  *
129  * another special case : 1 x1 x2 x3 + 1 x1 x2 x4 + 1 x5 x6 <= 1 ;
130  * which "equals" a pseudoboolean constraint: 2 x1 + 2 x2 + 1 x3 + 1 x4 + 1 x5 x6 <= 5 ;
131  * which "equals" a pseudoboolean constraint: 4 x1 + 4 x2 + 2 x3 + 2 x4 + 1 x5 + 1 x6 <= 10 ;
132  *
133  * another special case : 1 x1 x2 + 1 x1 x3 + 2 x4 x5 <= 3 ;
134  * which "equals" a pseudoboolean constraint: 2 x1 + 1 x2 + 1 x3 + 2 x4 x5 <= 5 ;
135  * which "equals" a pseudoboolean constraint: 2 x1 + 1 x2 + 1 x3 + 1 x4 + 1 x5 <= 5 ;
136  */
137 /* @todo - in and-constraint better count nfixed zeros in both directions and maybe nfixedones for better propagation
138  *
139  * - do better conflict analysis by choosing the earliest fixed variable which led to a conflict instead of maybe
140  * best coefficient or create more conflicts by using all to zero fixed variables one by one
141  *
142  * - how to make sure that we aggregate in a right way, when aggregating a resultant and a "normal" variable,
143  * maybe add in SCIPaggregateVars a check for original variables, to prefer them if the variable type is the
144  * same; probably it would be better too if we would aggregate two resultants that the one with less variables
145  * inside the and-constraint will stay active
146  *
147  * @note since product resultants are artificial, we do not care for their solution value, but this can lead to fixation
148  * of the resultant not representing the product, in 'optimization mode' we do not care, but this might make
149  * solution debugging complicated
150  */
151 
152 /** and-constraint data object */
153 struct ConsAndData
154 {
155  SCIP_CONS* cons; /**< pointer to the and-constraint of this 'term' of variables */
156  SCIP_CONS* origcons; /**< pointer to the original and-constraint of this 'term' of variables
157  * only after problem was transformed, NULL otherwise */
158  SCIP_VAR** vars; /**< all and-constraint variables */
159  int nvars; /**< number of all and-constraint variables */
160  int svars; /**< size for all and-constraint variables */
161  SCIP_VAR** newvars; /**< new variables in this presolving round */
162  int nnewvars; /**< number of new variables in this presolving round */
163  int snewvars; /**< size of new variables in this presolving round */
164  int noriguses; /**< how often is this data in used by original constraints */
165  int nuses; /**< how often is this data in used by transformed constraints */
166  unsigned int istransformed:1; /**< is transformed data active */
167  unsigned int isoriginal:1; /**< is original data active */
168 };
169 typedef struct ConsAndData CONSANDDATA;
171 /** constraint data for pseudoboolean constraints */
172 struct SCIP_ConsData
173 {
174  SCIP_Real lhs; /**< left hand side of constraint */
175  SCIP_Real rhs; /**< right hand side of constraint */
176 
177  SCIP_CONS* lincons; /**< linear constraint which represents this pseudoboolean constraint */
178  SCIP_LINEARCONSTYPE linconstype; /**< type of linear constraint which represents this pseudoboolean constraint */
179  int nlinvars; /**< number of linear variables (without and-resultants) */
180 
181  CONSANDDATA** consanddatas; /**< array of and-constraints-data-objects sorted after index of
182  * and-resultant of corresponding and-constraint */
183  SCIP_Real* andcoefs; /**< array of coefficients for and-constraints of
184  * and-constraints-data-objects
185  * (changes during presolving, needs to be updated in every presolving
186  * round) */
187  SCIP_Bool* andnegs; /**< array of negation status for and-constraints of
188  * and-constraints-data-objects
189  * (changes during presolving, needs to be updated in every presolving
190  * round) */
191  int nconsanddatas; /**< number of and-constraints-data-objects */
192  int sconsanddatas; /**< size of and-constraints-data-objects array */
193 
194  SCIP_VAR* intvar; /**< a artificial variable which was added only for the objective function,
195  * if this variable is not NULL this constraint (without this integer
196  * variable) describes the objective function */
197 
198  SCIP_VAR* indvar; /**< indicator variable if it's a soft constraint, or NULL */
199  SCIP_Real weight; /**< weight of the soft constraint, if it is one */
200 
201  unsigned int issoftcons:1; /**< is this a soft constraint */
202  unsigned int changed:1; /**< was constraint changed? */
203  unsigned int propagated:1; /**< is constraint already propagated? */
204  unsigned int presolved:1; /**< is constraint already presolved? */
205  unsigned int cliquesadded:1; /**< were the cliques of the constraint already extracted? */
206  unsigned int upgradetried:1; /**< was constraint upgrading already tried */
207 };
208 
209 /** constraint handler data */
210 struct SCIP_ConshdlrData
211 {
212  CONSANDDATA** allconsanddatas; /**< array of all and-constraint data objects inside the whole problem,
213  * created via this constraint handler */
214  int nallconsanddatas; /**< number of all and-constraint data objects inside the whole problem,
215  * created via this constraint handler */
216  int sallconsanddatas; /**< size of all and-constraint data objects inside the whole problem,
217  * created via this constraint handler */
218  SCIP_HASHTABLE* hashtable; /**< hash table for all and-constraint data objects */
219  int hashtablesize; /**< size for hash table for all and-constraint data objects */
220 
221  SCIP_HASHMAP* hashmap; /**< hash map for mapping all resultant to and-constraint */
222  int hashmapsize; /**< size for hash map for mapping all resultant to and-constraint */
223 
224  SCIP_Bool decomposenormalpbcons;/**< decompose the pseudo boolean constraint into a "linear" constraint and "and" constraints */
225  SCIP_Bool decomposeindicatorpbcons;/**< decompose the indicator pseudo boolean constraint into a "linear" constraint and "and" constraints */
226  SCIP_Bool inithashmapandtable;/**< flag to store if the hashmap and -table is initialized */
227  int nlinconss; /**< for counting number of created linear constraints */
228  int noriguses; /**< how many consanddata objects are used by original constraints */
229 };
230 
231 /*
232  * Local methods
233  */
234 
235 
236 /** comparison method for sorting consanddatas according to the index of their corresponding resultant variables, if a
237  * consanddata object is delete it is handled like it has an inactive resultant, so this will be put in front while
238  * sorting
239  */
240 static
241 SCIP_DECL_SORTPTRCOMP(resvarCompWithInactive)
242 {
243  CONSANDDATA* consanddata1;
244  CONSANDDATA* consanddata2;
245 
246  consanddata1 = (CONSANDDATA*)elem1;
247  consanddata2 = (CONSANDDATA*)elem2;
248 
249  /* check if and constraint data object is still valid */
250  if( !consanddata1->istransformed )
251  {
252  if( !consanddata2->istransformed )
253  {
254  return 0;
255  }
256  else
257  return -1;
258  }
259  else if( !consanddata2->istransformed )
260  return +1;
261 
262  assert(consanddata1->cons != NULL);
263  assert(consanddata2->cons != NULL);
264 
265  /* check if and constraint is still active */
266  if( SCIPconsIsDeleted(consanddata1->cons) )
267  {
268  if( SCIPconsIsDeleted(consanddata2->cons) )
269  {
270  return 0;
271  }
272  else
273  return -1;
274  }
275  else if( SCIPconsIsDeleted(consanddata2->cons) )
276  return +1;
277  else
278  {
279  SCIP_VAR* var1;
280  SCIP_VAR* var2;
281 
282  /* hack with setting the first pointer to NULL */
283  var1 = SCIPgetResultantAnd(NULL, consanddata1->cons);
284  var2 = SCIPgetResultantAnd(NULL, consanddata2->cons);
285 
286  assert(var1 != NULL);
287  assert(var2 != NULL);
288 
289  if( SCIPvarGetIndex(var1) < SCIPvarGetIndex(var2) )
290  return -1;
291  else if( SCIPvarGetIndex(var1) > SCIPvarGetIndex(var2) )
292  return +1;
293  else
294  {
295  assert(var1 == var2);
296  return 0;
297  }
298  }
299 }
300 
301 /** gets the key of the given element */
302 static
303 SCIP_DECL_HASHGETKEY(hashGetKeyAndConsDatas)
304 { /*lint --e{715}*/
305  /* the key is the element itself */
306  return elem;
307 }
308 
309 /** returns TRUE iff both keys are equal; two non-linear terms are equal if they have the same variables */
310 static
311 SCIP_DECL_HASHKEYEQ(hashKeyEqAndConsDatas)
312 {
313 #ifndef NDEBUG
314  SCIP* scip;
315 #endif
316  CONSANDDATA* cdata1;
317  CONSANDDATA* cdata2;
318  int v;
319 
320  cdata1 = (CONSANDDATA*)key1;
321  cdata2 = (CONSANDDATA*)key2;
322 
323 #ifndef NDEBUG
324  scip = (SCIP*)userptr;
325 #endif
326  assert(scip != NULL);
327  assert(cdata1 != NULL);
328  assert(cdata2 != NULL);
329  assert(cdata1->vars != NULL);
330  assert(cdata1->nvars > 1);
331  assert(cdata2->vars != NULL);
332  assert(cdata2->nvars > 1);
333 
334 #ifndef NDEBUG
335  /* check that cdata1 variables are sorted */
336  for( v = cdata1->nvars - 1; v > 0; --v )
337  assert(SCIPvarGetIndex(cdata1->vars[v]) >= SCIPvarGetIndex(cdata1->vars[v - 1]));
338  /* check that cdata2 variables are sorted */
339  for( v = cdata2->nvars - 1; v > 0; --v )
340  assert(SCIPvarGetIndex(cdata2->vars[v]) >= SCIPvarGetIndex(cdata2->vars[v - 1]));
341 #endif
342 
343  /* checks trivial case */
344  if( cdata1->nvars != cdata2->nvars )
345  return FALSE;
346 
347  /* checks trivial case */
348  if( cdata1->cons != NULL && cdata2->cons != NULL && cdata1->cons != cdata2->cons )
349  return FALSE;
350 
351  /* check each variable in both cdatas for equality */
352  for( v = cdata1->nvars - 1; v >= 0; --v )
353  {
354  assert(cdata1->vars[v] != NULL);
355  assert(cdata2->vars[v] != NULL);
356 
357  /* tests if variables are equal */
358  if( cdata1->vars[v] != cdata2->vars[v] )
359  {
360  assert(SCIPvarCompare(cdata1->vars[v], cdata2->vars[v]) == 1 ||
361  SCIPvarCompare(cdata1->vars[v], cdata2->vars[v]) == -1);
362  return FALSE;
363  }
364  assert(SCIPvarCompare(cdata1->vars[v], cdata2->vars[v]) == 0);
365  }
366 
367  return TRUE;
368 }
369 
370 /** returns the hash value of the key */
371 static
372 SCIP_DECL_HASHKEYVAL(hashKeyValAndConsDatas)
373 { /*lint --e{715}*/
374  CONSANDDATA* cdata;
375  int minidx;
376  int mididx;
377  int maxidx;
378 
379  cdata = (CONSANDDATA*)key;
380 
381  assert(cdata != NULL);
382  assert(cdata->vars != NULL);
383  assert(cdata->nvars > 1);
384 #ifndef NDEBUG
385  {
386  /* check that these variables are sorted */
387  int v;
388  for( v = cdata->nvars - 1; v > 0; --v )
389  assert(SCIPvarGetIndex(cdata->vars[v]) >= SCIPvarGetIndex(cdata->vars[v - 1]));
390  }
391 #endif
392 
393  minidx = SCIPvarGetIndex(cdata->vars[0]);
394  mididx = SCIPvarGetIndex(cdata->vars[cdata->nvars / 2]);
395  maxidx = SCIPvarGetIndex(cdata->vars[cdata->nvars - 1]);
396  assert(minidx >= 0 && minidx <= maxidx);
397 
398  return SCIPhashFour(cdata->nvars, minidx, mididx, maxidx);
399 }
400 
401 /** initializes the hashmap and -table used in this constraint handler data for artificial variables and specific
402  * and-constraint data objects
403  */
404 static
406  SCIP*const scip, /**< SCIP data structure */
407  SCIP_CONSHDLRDATA** conshdlrdata /**< pointer to store the constraint handler data */
408  )
409 {
410  if( ((*conshdlrdata)->inithashmapandtable) )
411  {
412  assert((*conshdlrdata)->hashtable != NULL);
413  assert((*conshdlrdata)->hashmap != NULL);
414 
415  return SCIP_OKAY;
416  }
417 
418  assert((*conshdlrdata)->hashtable == NULL);
419  assert((*conshdlrdata)->hashmap == NULL);
420 
421  /* create a hash table for and-constraint data objects */
422  (*conshdlrdata)->hashtablesize = HASHSIZE_PSEUDOBOOLEANNONLINEARTERMS;
423  SCIP_CALL( SCIPhashtableCreate(&((*conshdlrdata)->hashtable), SCIPblkmem(scip), (*conshdlrdata)->hashtablesize,
424  hashGetKeyAndConsDatas, hashKeyEqAndConsDatas, hashKeyValAndConsDatas, (void*) scip) );
425 
426  /* create a hash table for and-resultant to and-constraint data objects */
427  (*conshdlrdata)->hashmapsize = HASHSIZE_PSEUDOBOOLEANNONLINEARTERMS;
428  SCIP_CALL( SCIPhashmapCreate(&((*conshdlrdata)->hashmap), SCIPblkmem(scip), (*conshdlrdata)->hashmapsize) );
429 
430  (*conshdlrdata)->inithashmapandtable = TRUE;
431 
432  return SCIP_OKAY;
433 }
434 
435 /** creates constraint handler data for pseudo boolean constraint handler */
436 static
438  SCIP*const scip, /**< SCIP data structure */
439  SCIP_CONSHDLRDATA** conshdlrdata /**< pointer to store the constraint handler data */
440  )
441 {
442  assert(scip != NULL);
443  assert(conshdlrdata != NULL);
444 
445  SCIP_CALL( SCIPallocBlockMemory(scip, conshdlrdata) );
446 
447  (*conshdlrdata)->allconsanddatas = NULL;
448  (*conshdlrdata)->nallconsanddatas = 0;
449  (*conshdlrdata)->sallconsanddatas = 10;
450 
451  SCIP_CALL( SCIPallocBlockMemoryArray(scip, &((*conshdlrdata)->allconsanddatas), (*conshdlrdata)->sallconsanddatas ) );
452 
453  /* set hashmap and -table to NULL, mark them as uninitialized */
454  (*conshdlrdata)->inithashmapandtable = FALSE;
455  (*conshdlrdata)->hashtable = NULL;
456  (*conshdlrdata)->hashtablesize = 0;
457  (*conshdlrdata)->hashmap = NULL;
458  (*conshdlrdata)->hashmapsize = 0;
459 
460  /* for constraint names count number of created constraints */
461  (*conshdlrdata)->nlinconss = 0;
462 
463  /* initializes how many consanddata objects are used by original constraints */
464  (*conshdlrdata)->noriguses = 0;
465 
466  return SCIP_OKAY;
467 }
468 
469 
470 /** frees constraint handler data for pseudo boolean constraint handler */
471 static
473  SCIP*const scip, /**< SCIP data structure */
474  SCIP_CONSHDLRDATA** conshdlrdata /**< pointer to the constraint handler data */
475  )
476 {
477  assert(scip != NULL);
478  assert(conshdlrdata != NULL);
479  assert(*conshdlrdata != NULL);
480  assert((*conshdlrdata)->nallconsanddatas == 0);
481 
482  /* free hash table if necessary */
483  if( (*conshdlrdata)->inithashmapandtable )
484  {
485  SCIPhashmapFree(&((*conshdlrdata)->hashmap));
486  (*conshdlrdata)->hashmapsize = 0;
487  SCIPhashtableFree(&((*conshdlrdata)->hashtable));
488  (*conshdlrdata)->hashtablesize = 0;
489  }
490  else
491  {
492  assert((*conshdlrdata)->hashmap == NULL);
493  assert((*conshdlrdata)->hashtable == NULL);
494  }
495  (*conshdlrdata)->inithashmapandtable = FALSE;
496 
497  /* clear array for all consanddata objects */
498  SCIPfreeBlockMemoryArray(scip, &((*conshdlrdata)->allconsanddatas), (*conshdlrdata)->sallconsanddatas );
499 
500  (*conshdlrdata)->allconsanddatas = NULL;
501  (*conshdlrdata)->nallconsanddatas = 0;
502  (*conshdlrdata)->sallconsanddatas = 0;
503 
504  SCIPfreeBlockMemory(scip, conshdlrdata);
505 
506  return SCIP_OKAY;
507 }
508 
509 /** gets number of variables in linear constraint */
510 static
512  SCIP*const scip, /**< SCIP data structure */
513  SCIP_CONS*const cons, /**< linear constraint */
514  SCIP_LINEARCONSTYPE const constype, /**< linear constraint type */
515  int*const nvars /**< pointer to store number variables of linear constraint */
516  )
517 {
518  assert(scip != NULL);
519  assert(cons != NULL);
520  assert(nvars != NULL);
521 
522  /* determine for each special linear constranit all variables and coefficients */
523  switch( constype )
524  {
526  *nvars = SCIPgetNVarsLinear(scip, cons);
527  break;
529  *nvars = SCIPgetNVarsLogicor(scip, cons);
530  break;
532  *nvars = SCIPgetNVarsKnapsack(scip, cons);
533  break;
535  *nvars = SCIPgetNVarsSetppc(scip, cons);
536  break;
537 #ifdef WITHEQKNAPSACK
538  case SCIP_LINEARCONSTYPE_EQKNAPSACK:
539  *nvars = SCIPgetNVarsEQKnapsack(scip, cons);
540  break;
541 #endif
543  default:
544  SCIPerrorMessage("unknown linear constraint type\n");
545  return SCIP_INVALIDDATA;
546  }
547 
548  return SCIP_OKAY;
549 }
550 
551 
552 /** gets sides of linear constraint */
553 static
555  SCIP*const scip, /**< SCIP data structure */
556  SCIP_CONS*const cons, /**< linear constraint */
557  SCIP_LINEARCONSTYPE const constype, /**< linear constraint type */
558  SCIP_Real*const lhs, /**< pointer to store left hand side of linear constraint */
559  SCIP_Real*const rhs /**< pointer to store right hand side of linear constraint */
560  )
561 {
562  SCIP_SETPPCTYPE type;
563 
564  switch( constype )
565  {
567  *lhs = SCIPgetLhsLinear(scip, cons);
568  *rhs = SCIPgetRhsLinear(scip, cons);
569  break;
571  *lhs = 1.0;
572  *rhs = SCIPinfinity(scip);
573  break;
575  *lhs = -SCIPinfinity(scip);
576  *rhs = SCIPgetCapacityKnapsack(scip, cons);
577  break;
579  type = SCIPgetTypeSetppc(scip, cons);
580 
581  switch( type )
582  {
584  *lhs = 1.0;
585  *rhs = 1.0;
586  break;
588  *lhs = -SCIPinfinity(scip);
589  *rhs = 1.0;
590  break;
592  *lhs = 1.0;
593  *rhs = SCIPinfinity(scip);
594  break;
595  default:
596  SCIPerrorMessage("unknown setppc type\n");
597  return SCIP_INVALIDDATA;
598  }
599  break;
600 #ifdef WITHEQKNAPSACK
601  case SCIP_LINEARCONSTYPE_EQKNAPSACK:
602  *lhs = SCIPgetCapacityEQKnapsack(scip, cons);
603  *rhs = *lhs;
604  break;
605 #endif
607  default:
608  SCIPerrorMessage("unknown linear constraint type\n");
609  return SCIP_INVALIDDATA;
610  }
611 
612  return SCIP_OKAY;
613 }
614 
615 /** gets variables and coefficients of linear constraint */
616 static
618  SCIP*const scip, /**< SCIP data structure */
619  SCIP_CONS*const cons, /**< linear constraint */
620  SCIP_LINEARCONSTYPE const constype, /**< linear constraint type */
621  SCIP_VAR**const vars, /**< array to store sorted (after indices) variables of linear constraint */
622  SCIP_Real*const coefs, /**< array to store coefficient of linear constraint, or NULL */
623  int*const nvars /**< pointer to store number variables of linear constraint */
624  )
625 {
626  SCIP_VAR** linvars;
627  int v;
628 
629  assert(scip != NULL);
630  assert(cons != NULL);
631  assert(vars != NULL);
632  assert(nvars != NULL);
633 
634  /* determine for each special linear constrait all variables and coefficients */
635  switch( constype )
636  {
638  {
639  SCIP_Real* lincoefs;
640 
641  *nvars = SCIPgetNVarsLinear(scip, cons);
642  linvars = SCIPgetVarsLinear(scip, cons);
643 
644  if( coefs != NULL )
645  {
646  lincoefs = SCIPgetValsLinear(scip, cons);
647 
648  for( v = 0; v < *nvars; ++v )
649  {
650  vars[v] = linvars[v];
651  coefs[v] = lincoefs[v];
652  }
653  }
654  else
655  {
656  for( v = 0; v < *nvars; ++v )
657  vars[v] = linvars[v];
658  }
659 
660  break;
661  }
663  *nvars = SCIPgetNVarsLogicor(scip, cons);
664  linvars = SCIPgetVarsLogicor(scip, cons);
665  assert( linvars != NULL );
666 
667  if( coefs != NULL )
668  {
669  for( v = 0; v < *nvars; ++v )
670  {
671  vars[v] = linvars[v];
672  coefs[v] = 1.0;
673  }
674  }
675  else
676  {
677  for( v = 0; v < *nvars; ++v )
678  vars[v] = linvars[v];
679  }
680 
681  break;
683  {
684  SCIP_Longint* weights;
685 
686  *nvars = SCIPgetNVarsKnapsack(scip, cons);
687  linvars = SCIPgetVarsKnapsack(scip, cons);
688  assert( linvars != NULL );
689 
690  if( coefs != NULL )
691  {
692  weights = SCIPgetWeightsKnapsack(scip, cons);
693 
694  for( v = 0; v < *nvars; ++v )
695  {
696  vars[v] = linvars[v];
697  coefs[v] = (SCIP_Real) weights[v];
698  }
699  }
700  else
701  {
702  for( v = 0; v < *nvars; ++v )
703  vars[v] = linvars[v];
704  }
705 
706  break;
707  }
709  *nvars = SCIPgetNVarsSetppc(scip, cons);
710  linvars = SCIPgetVarsSetppc(scip, cons);
711  assert( linvars != NULL );
712 
713  if( coefs != NULL )
714  {
715  for( v = 0; v < *nvars; ++v )
716  {
717  vars[v] = linvars[v];
718  coefs[v] = 1.0;
719  }
720  }
721  else
722  {
723  for( v = 0; v < *nvars; ++v )
724  vars[v] = linvars[v];
725  }
726 
727  break;
728 #ifdef WITHEQKNAPSACK
729  case SCIP_LINEARCONSTYPE_EQKNAPSACK:
730  {
731  SCIP_Longint* weights;
732 
733  *nvars = SCIPgetNVarsEQKnapsack(scip, cons);
734  linvars = SCIPgetVarsEQKnapsack(scip, cons);
735  assert( linvars != NULL );
736 
737  if( coefs != NULL )
738  {
739  weights = SCIPgetWeightsEQKnapsack(scip, cons);
740 
741  for( v = 0; v < *nvars; ++v )
742  {
743  vars[v] = linvars[v];
744  coefs[v] = (SCIP_Real) weights[v];
745  }
746  }
747  else
748  {
749  for( v = 0; v < *nvars; ++v )
750  vars[v] = linvars[v];
751  }
752 
753  break;
754  }
755 #endif
757  default:
758  SCIPerrorMessage("unknown linear constraint type\n");
759  return SCIP_INVALIDDATA;
760  }
761 
762  /* sort variables after indices */
763  if( coefs != NULL )
764  {
765  SCIPsortPtrReal((void**)vars, coefs, SCIPvarComp, *nvars);
766  }
767  else
768  {
769  SCIPsortPtr((void**)vars, SCIPvarComp, *nvars);
770  }
771 
772  return SCIP_OKAY;
773 }
774 
775 /** calculate all not artificial linear variables and all artificial and-resultants which will be ordered like the
776  * 'consanddatas' such that the and-resultant of the and-constraint is the and-resultant in the 'andress' array
777  * afterwards
778  */
779 static
781  SCIP*const scip, /**< SCIP data structure */
782  SCIP_CONS*const cons, /**< pseudoboolean constraint */
783  SCIP_VAR**const vars, /**< all variables of linear constraint */
784  SCIP_Real*const coefs, /**< all coefficients of linear constraint, or NULL */
785  int const nvars, /**< number of all variables of linear constraint */
786  SCIP_VAR**const linvars, /**< array to store not and-resultant variables of linear constraint, or NULL */
787  SCIP_Real*const lincoefs, /**< array to store coefficients of not and-resultant variables of linear
788  * constraint, or NULL */
789  int*const nlinvars, /**< pointer to store number of not and-resultant variables, or NULL */
790  SCIP_VAR**const andress, /**< array to store and-resultant variables of linear constraint, or NULL */
791  SCIP_Real*const andcoefs, /**< array to store coefficients of and-resultant variables of linear
792  * constraint, or NULL */
793  SCIP_Bool*const andnegs, /**< array to store negation status of and-resultant variables of linear
794  * constraint, or NULL */
795  int*const nandress /**< pointer to store number of and-resultant variables, or NULL */
796  )
797 {
798  SCIP_CONSHDLR* conshdlr;
799  SCIP_CONSHDLRDATA* conshdlrdata;
800  int v;
801 
802  assert(scip != NULL);
803  assert(cons != NULL);
804  assert(vars != NULL);
805  assert((linvars != NULL) == (nlinvars != NULL));
806  assert((andress == NULL) || (nandress != NULL));
807  assert((andcoefs != NULL) == (andnegs != NULL));
808  assert((coefs != NULL) == ((lincoefs != NULL) || (andcoefs != NULL)));
809  assert(linvars != NULL || andress != NULL);
810 
811  if( nlinvars != NULL )
812  *nlinvars = 0;
813  if( nandress != NULL )
814  *nandress = 0;
815 
816  conshdlr = SCIPconsGetHdlr(cons);
817  assert(conshdlr != NULL);
818  conshdlrdata = SCIPconshdlrGetData(conshdlr);
819  assert(conshdlrdata != NULL);
820  assert(conshdlrdata->hashmap != NULL);
821 
822  /* @note it is necessary that the linear constraint is merged (not needed for negated variables) and sorted after
823  * indices
824  */
825 
826 #ifndef NDEBUG
827  /* check that old variables are sorted */
828  for( v = nvars - 1; v > 0; --v )
829  assert(SCIPvarGetIndex(vars[v]) >= SCIPvarGetIndex(vars[v - 1]));
830 #endif
831 
832  /* split variables into original and artificial variables */
833  for( v = 0; v < nvars; ++v )
834  {
835  SCIP_Bool hashmapentryexists;
836  SCIP_VAR* hashmapvar;
837 
838  assert(vars[v] != NULL);
839 
840  hashmapentryexists = SCIPhashmapExists(conshdlrdata->hashmap, (void*)(vars[v]));
841 
842  if( !hashmapentryexists && SCIPvarIsNegated(vars[v]) )
843  {
844  hashmapvar = SCIPvarGetNegationVar(vars[v]);
845  hashmapentryexists = SCIPhashmapExists(conshdlrdata->hashmap, (void*)(hashmapvar));
846  }
847  else
848  hashmapvar = vars[v];
849 
850  /* if and resultant is not a resultant anymore (meaning the corresponding and-constraint was deleted/upgraded),
851  * correct the flag and count this variable as normal linear variable
852  */
853  if( hashmapentryexists )
854  {
855  if( !SCIPconsIsOriginal(cons) )
856  {
857  CONSANDDATA* consanddata = (CONSANDDATA*) SCIPhashmapGetImage(conshdlrdata->hashmap, (void*)(hashmapvar));
858  assert(consanddata != NULL);
859 
860  hashmapentryexists = (consanddata->istransformed);
861 
862  if( hashmapentryexists )
863  {
864  assert(consanddata->cons != NULL);
865  hashmapentryexists = !SCIPconsIsDeleted(consanddata->cons);
866  }
867  }
868  }
869 
870  if( !hashmapentryexists && linvars != NULL && nlinvars != NULL )
871  {
872  linvars[*nlinvars] = vars[v];
873  if( lincoefs != NULL )
874  {
875  assert(coefs != NULL);
876  lincoefs[*nlinvars] = coefs[v];
877  }
878  ++(*nlinvars);
879  }
880  else if( hashmapentryexists && nandress != NULL )
881  {
882  if( andress != NULL )
883  {
884  andress[*nandress] = hashmapvar;
885 
886  if( andcoefs != NULL )
887  {
888  assert(andnegs != NULL);
889  assert(coefs != NULL);
890  andcoefs[*nandress] = coefs[v];
891  andnegs[*nandress] = (vars[v] != hashmapvar);
892  }
893  }
894  ++(*nandress);
895  }
896  }
897 
898  /* @todo try to avoid sorting here */
899  if( andress != NULL && nandress != NULL )
900  {
901  /* sort and resultants by their variable index */
902  if( andcoefs != NULL )
903  {
904  assert(andnegs != NULL);
905  SCIPsortPtrRealBool((void**)andress, andcoefs, andnegs, SCIPvarComp, *nandress);
906  }
907  else
908  {
909  SCIPsortPtr((void**)andress, SCIPvarComp, *nandress);
910  }
911  }
912 
913  return SCIP_OKAY;
914 }
915 
916 
917 #ifdef CHECK_CONSISTENCY
918 /** check constraint consistency */
919 static
921  SCIP*const scip, /**< SCIP data structure */
922  SCIP_CONS*const cons /**< pseudoboolean constraint */
923  )
924 {
925  SCIP_CONSDATA* consdata;
926  SCIP_VAR** vars;
927  SCIP_Real* coefs;
928  int nvars;
929  SCIP_VAR** linvars;
930  SCIP_Real* lincoefs;
931  int nlinvars;
932  SCIP_VAR** andress;
933  SCIP_Real* andcoefs;
934  SCIP_Bool* andnegs;
935  int nandress;
936  SCIP_Bool* alreadyfound;
937  SCIP_VAR* res;
938  int c;
939  int v;
940  SCIP_Real newlhs;
941  SCIP_Real newrhs;
942 
943  assert(scip != NULL);
944  assert(cons != NULL);
945 
946  if( SCIPgetStage(scip) == SCIP_STAGE_FREETRANS )
947  return;
948 
949  consdata = SCIPconsGetData(cons);
950  assert(consdata != NULL);
951 
952  /* check standard pointers and sizes */
953  assert(consdata->lincons != NULL);
954  assert(!SCIPconsIsDeleted(consdata->lincons));
955  assert(consdata->linconstype > SCIP_LINEARCONSTYPE_INVALIDCONS);
956  assert(consdata->consanddatas != NULL);
957  assert(consdata->nconsanddatas > 0);
958  assert(consdata->nconsanddatas <= consdata->sconsanddatas);
959 
960  /* get sides of linear constraint */
961  SCIP_CALL_ABORT( getLinearConsSides(scip, consdata->lincons, consdata->linconstype, &newlhs, &newrhs) );
962  assert(!SCIPisInfinity(scip, newlhs));
963  assert(!SCIPisInfinity(scip, -newrhs));
964  assert(SCIPisLE(scip, newlhs, newrhs));
965  assert(SCIPisEQ(scip, newrhs, consdata->rhs) || SCIPisEQ(scip, newrhs, -consdata->lhs));
966  assert(SCIPisEQ(scip, newlhs, consdata->lhs) || SCIPisEQ(scip, newlhs, -consdata->rhs));
967 
968  /* check number of linear variables */
969  SCIP_CALL_ABORT( getLinearConsNVars(scip, consdata->lincons, consdata->linconstype, &nvars) );
970  assert(nvars == consdata->nlinvars + consdata->nconsanddatas);
971 
972  /* get temporary memory */
973  SCIP_CALL_ABORT( SCIPallocBufferArray(scip, &vars, nvars) );
974  SCIP_CALL_ABORT( SCIPallocBufferArray(scip, &coefs, nvars) );
975  SCIP_CALL_ABORT( SCIPallocBufferArray(scip, &linvars, nvars) );
976  SCIP_CALL_ABORT( SCIPallocBufferArray(scip, &lincoefs, nvars) );
977  SCIP_CALL_ABORT( SCIPallocBufferArray(scip, &andress, nvars) );
978  SCIP_CALL_ABORT( SCIPallocBufferArray(scip, &andcoefs, nvars) );
979  SCIP_CALL_ABORT( SCIPallocBufferArray(scip, &andnegs, nvars) );
980  SCIP_CALL_ABORT( SCIPallocClearBufferArray(scip, &alreadyfound, nvars) );
981 
982  /* get variables and coefficients */
983  SCIP_CALL_ABORT( getLinearConsVarsData(scip, consdata->lincons, consdata->linconstype, vars, coefs, &nvars) );
984  assert(nvars == 0 || (coefs != NULL));
985 
986  /* calculate all not artificial linear variables and all artificial and-resultants */
987  SCIP_CALL_ABORT( getLinVarsAndAndRess(scip, cons, vars, coefs, nvars, linvars, lincoefs, &nlinvars,
988  andress, andcoefs, andnegs, &nandress) );
989  assert(nlinvars == consdata->nlinvars);
990  assert(nandress == consdata->nconsanddatas);
991 
992  for( v = nandress - 1; v >= 0; --v )
993  {
994  SCIP_VAR* andresultant = andress[v];
995  int nfound = 0;
996 
997  for( c = consdata->nconsanddatas - 1; c >= 0; --c )
998  {
999  assert(consdata->consanddatas[c] != NULL);
1000  if( consdata->consanddatas[c]->cons != NULL )
1001  {
1002  res = SCIPgetResultantAnd(scip, consdata->consanddatas[c]->cons);
1003  assert(res != NULL);
1004 
1005  if( res == andresultant && consdata->andnegs[c] == andnegs[v] && consdata->andcoefs[c] == andcoefs[v] )
1006  {
1007  /* resultant should be either active or a negated variable of an active one */
1009  assert(!alreadyfound[c]);
1010 
1011  /* all and-resultants should be merged, so it is only allowed that each variable exists one time */
1012  alreadyfound[c] = TRUE;
1013  ++nfound;
1014  break;
1015  }
1016  }
1017  }
1018  assert(nfound == 1);
1019  }
1020 
1021  for( c = consdata->nconsanddatas - 1; c >= 0; --c )
1022  {
1023  assert(alreadyfound[c]);
1024  }
1025 
1026  /* free temporary memory */
1027  SCIPfreeBufferArray(scip, &alreadyfound);
1028  SCIPfreeBufferArray(scip, &andnegs);
1029  SCIPfreeBufferArray(scip, &andcoefs);
1030  SCIPfreeBufferArray(scip, &andress);
1031  SCIPfreeBufferArray(scip, &lincoefs);
1032  SCIPfreeBufferArray(scip, &linvars);
1033  SCIPfreeBufferArray(scip, &coefs);
1034  SCIPfreeBufferArray(scip, &vars);
1035 }
1036 #else
1037 #define checkConsConsistency(scip, cons) /**/
1038 #endif
1039 
1040 
1041 /** transforming transformed consanddata object back to original space, if an corresponding original constraint exists,
1042  * also clearing all transformed data, i.e. releasing transformed variables
1043  */
1044 static
1046  SCIP*const scip, /**< SCIP data structure */
1047  CONSANDDATA* consanddata, /**< consanddata object */
1048  SCIP_CONSHDLRDATA* conshdlrdata /**< constraint handler data */
1049  )
1050 {
1051  SCIP_VAR** tmpvars;
1052  SCIP_Bool origdata;
1053  int ntmpvars;
1054  int v;
1055 
1056  assert(scip != NULL);
1057  assert(consanddata != NULL);
1058  assert(conshdlrdata != NULL);
1059 
1060  origdata = TRUE;
1061 
1062  tmpvars = consanddata->vars;
1063  ntmpvars = consanddata->nvars;
1064 
1065  /* release all transformed variables */
1066  for( v = ntmpvars - 1; v >= 0; --v )
1067  {
1068  assert(tmpvars[v] != NULL);
1069  if( SCIPvarIsTransformed(tmpvars[v]) )
1070  {
1071  SCIP_CALL( SCIPreleaseVar(scip, &tmpvars[v]) );
1072  origdata = FALSE;
1073  }
1074  }
1075 
1076  tmpvars = consanddata->newvars;
1077  ntmpvars = consanddata->nnewvars;
1078 
1079  /* release all variables */
1080  for( v = ntmpvars - 1; v >= 0; --v )
1081  {
1082  assert(tmpvars[v] != NULL);
1083  if( SCIPvarIsTransformed(tmpvars[v]) )
1084  {
1085  SCIP_CALL( SCIPreleaseVar(scip, &tmpvars[v]) );
1086  origdata = FALSE;
1087  }
1088  }
1089 
1090  /* reinstall original data */
1091  if( !origdata || consanddata->nvars == 0 )
1092  {
1093  SCIPfreeBlockMemoryArrayNull(scip, &(consanddata->vars), consanddata->svars);
1094  SCIPfreeBlockMemoryArrayNull(scip, &(consanddata->newvars), consanddata->snewvars);
1095 
1096  consanddata->nuses = 0;
1097  consanddata->nvars = 0;
1098  consanddata->svars = 0;
1099  consanddata->nnewvars = 0;
1100  consanddata->snewvars = 0;
1101  consanddata->istransformed = FALSE;
1102 
1103  if( consanddata->noriguses > 0 )
1104  {
1105  assert(consanddata->origcons != NULL);
1106  assert(consanddata->isoriginal);
1107 
1108  assert(SCIPgetNVarsAnd(scip, consanddata->origcons) > 0);
1109  assert(SCIPgetVarsAnd(scip, consanddata->origcons) != NULL);
1110  consanddata->nvars = SCIPgetNVarsAnd(scip, consanddata->origcons);
1111  consanddata->svars = consanddata->nvars;
1112 
1113  if( consanddata->nvars > 0 )
1114  {
1115  SCIP_VAR** andvars = SCIPgetVarsAnd(scip, consanddata->origcons);
1116 
1117  SCIP_CALL( SCIPduplicateBlockMemoryArray(scip, &(consanddata->vars), andvars, consanddata->nvars) );
1118 
1119  /* sort variables */
1120  SCIPsortPtr((void**)(consanddata->vars), SCIPvarComp, consanddata->nvars);
1121  }
1122 
1123  /* check that the hash map and tabkle are still having all information */
1124  if( conshdlrdata->inithashmapandtable )
1125  {
1126  assert(conshdlrdata->hashmap != NULL);
1127  assert(conshdlrdata->hashtable != NULL);
1128  assert(SCIPgetResultantAnd(scip, consanddata->origcons) != NULL);
1129  assert(SCIPhashtableExists(conshdlrdata->hashtable, (void*)consanddata));
1130  assert(consanddata == (CONSANDDATA*)(SCIPhashtableRetrieve(conshdlrdata->hashtable, (void*)consanddata)));
1131  assert(SCIPhashmapExists(conshdlrdata->hashmap, (void*)SCIPgetResultantAnd(scip, consanddata->origcons)));
1132  assert(consanddata == (CONSANDDATA*)(SCIPhashmapGetImage(conshdlrdata->hashmap, (void*)SCIPgetResultantAnd(scip, consanddata->origcons))));
1133  }
1134  }
1135  else
1136  assert(consanddata->origcons == NULL);
1137  }
1138  else
1139  {
1140  assert(consanddata->nuses == 0);
1141  assert(consanddata->nnewvars == 0);
1142  assert(consanddata->snewvars == 0);
1143  assert(consanddata->newvars == NULL);
1144 
1145  consanddata->istransformed = FALSE;
1146 
1147  if( consanddata->noriguses > 0 )
1148  {
1149  assert(consanddata->origcons != NULL);
1150  assert(consanddata->nvars > 0);
1151  assert(consanddata->svars > 0);
1152  assert(consanddata->vars != NULL);
1153  assert(consanddata->isoriginal);
1154 
1155  /* check that the hash map and tabkle are still having all information */
1156  if( conshdlrdata->inithashmapandtable )
1157  {
1158  assert(conshdlrdata->hashmap != NULL);
1159  assert(conshdlrdata->hashtable != NULL);
1160  assert(SCIPgetResultantAnd(scip, consanddata->origcons) != NULL);
1161  assert(SCIPhashtableExists(conshdlrdata->hashtable, (void*)consanddata));
1162  assert(consanddata == (CONSANDDATA*)(SCIPhashtableRetrieve(conshdlrdata->hashtable, (void*)consanddata)));
1163  assert(SCIPhashmapExists(conshdlrdata->hashmap, (void*)SCIPgetResultantAnd(scip, consanddata->origcons)));
1164  assert(consanddata == (CONSANDDATA*)(SCIPhashmapGetImage(conshdlrdata->hashmap, (void*)SCIPgetResultantAnd(scip, consanddata->origcons))));
1165  }
1166  }
1167  }
1168 
1169  return SCIP_OKAY;
1170 }
1171 
1172 
1173 
1174 /** creates a pseudo boolean constraint data */
1175 static
1177  SCIP*const scip, /**< SCIP data structure */
1178  SCIP_CONSHDLR*const conshdlr, /**< pseudoboolean constraint handler */
1179  SCIP_CONSDATA** consdata, /**< pointer to linear constraint data */
1180  SCIP_CONS*const lincons, /**< linear constraint with artificial and-resultants representing this pseudoboolean constraint */
1181  SCIP_LINEARCONSTYPE const linconstype, /**< type of linear constraint */
1182  SCIP_CONS**const andconss, /**< array of and-constraints which occur in this pseudoboolean constraint */
1183  SCIP_Real*const andcoefs, /**< coefficients of and-constraints */
1184  SCIP_Bool*const andnegs, /**< negation status of and-constraints (or NULL, if no negated resultants) */
1185  int const nandconss, /**< number of and-constraints */
1186  SCIP_VAR*const indvar, /**< indicator variable if it's a soft constraint, or NULL */
1187  SCIP_Real const weight, /**< weight of the soft constraint, if it is one */
1188  SCIP_Bool const issoftcons, /**< is this a soft constraint */
1189  SCIP_VAR* const intvar, /**< a artificial variable which was added only for the objective function,
1190  * if this variable is not NULL this constraint (without this integer
1191  * variable) describes the objective function */
1192  SCIP_Real lhs, /**< left hand side of row */
1193  SCIP_Real rhs, /**< right hand side of row */
1194  SCIP_Bool check, /**< is the new constraint a check constraint? */
1195  SCIP_Bool transforming /**< are we called by CONSTRANS */
1196  )
1197 {
1198  SCIP_Bool transformed;
1199  int nvars;
1200 
1201  assert(scip != NULL);
1202  assert(conshdlr != NULL);
1203  assert(consdata != NULL);
1204  assert(lincons != NULL && linconstype > SCIP_LINEARCONSTYPE_INVALIDCONS);
1205  assert(nandconss == 0 || (andconss != NULL && andcoefs != NULL));
1206  assert(!issoftcons || (!SCIPisZero(scip, weight) && indvar != NULL));
1207 
1208  /* adjust right hand side */
1209  if( SCIPisInfinity(scip, rhs) )
1210  rhs = SCIPinfinity(scip);
1211  else if( SCIPisInfinity(scip, -rhs) )
1212  rhs = -SCIPinfinity(scip);
1213 
1214  /* adjust left hand side */
1215  if( SCIPisInfinity(scip, -lhs) )
1216  lhs = -SCIPinfinity(scip);
1217  else if( SCIPisInfinity(scip, lhs) )
1218  lhs = SCIPinfinity(scip);
1219 
1220  /* check left and right side */
1221  if( SCIPisGT(scip, lhs, rhs) )
1222  {
1223  SCIPerrorMessage("left hand side of pseudo boolean constraint greater than right hand side\n");
1224  SCIPerrorMessage(" -> lhs=%g, rhs=%g\n", lhs, rhs);
1225  return SCIP_INVALIDDATA;
1226  }
1227 
1228  transformed = SCIPisTransformed(scip);
1229 
1230  /* allocate memory for the constraint data */
1231  SCIP_CALL( SCIPallocBlockMemory(scip, consdata) );
1232 
1233  /* initialize the weights for soft constraints */
1234  (*consdata)->issoftcons = issoftcons;
1235  if( issoftcons )
1236  {
1237  (*consdata)->weight = weight;
1238  if( transformed )
1239  {
1240  SCIP_CALL( SCIPgetTransformedVar(scip, indvar, &((*consdata)->indvar)) );
1241  }
1242  else
1243  (*consdata)->indvar = indvar;
1244  }
1245  else
1246  (*consdata)->indvar = NULL;
1247 
1248  /* copy artificial integer variable if it exist */
1249  if( intvar != NULL )
1250  {
1251  if( transformed )
1252  {
1253  SCIP_CALL( SCIPgetTransformedVar(scip, intvar, &((*consdata)->intvar)) );
1254  }
1255  else
1256  (*consdata)->intvar = intvar;
1257  }
1258  else
1259  (*consdata)->intvar = NULL;
1260 
1261  /* copy linear constraint */
1262  (*consdata)->lincons = lincons;
1263  (*consdata)->linconstype = linconstype;
1264 
1265  /* get transformed linear constraint and capture it if necessary */
1266  if( transforming )
1267  {
1268  /* do not capture the and constraint when scip is in transformed mode; this automatically happens in
1269  * SCIPtransformCons()
1270  */
1271  SCIP_CALL( SCIPtransformCons(scip, (*consdata)->lincons, &((*consdata)->lincons)) );
1272  assert((*consdata)->lincons != NULL);
1273  }
1274 
1275  if( transforming || transformed )
1276  {
1277  assert(SCIPconsIsTransformed((*consdata)->lincons));
1278 
1279  /* we want to check all necessary transformed linear constraints */
1280  SCIP_CALL( SCIPsetConsChecked(scip, (*consdata)->lincons, check) );
1281  }
1282 
1283  /* get number of non-linear terms in pseudoboolean constraint */
1284  SCIP_CALL( getLinearConsNVars(scip, (*consdata)->lincons, (*consdata)->linconstype, &nvars) );
1285  (*consdata)->nlinvars = nvars - nandconss;
1286 
1287  /* copy and-constraints */
1288  if( nandconss > 0 )
1289  {
1290  SCIP_CONSHDLRDATA* conshdlrdata;
1291  SCIP_VAR** andress;
1292  int c;
1293 
1294  SCIP_CALL( SCIPallocBlockMemoryArray(scip, &((*consdata)->consanddatas), nandconss) );
1295  SCIP_CALL( SCIPduplicateBlockMemoryArray(scip, &((*consdata)->andcoefs), andcoefs, nandconss) );
1296  if( andnegs != NULL )
1297  {
1298  SCIP_CALL( SCIPduplicateBlockMemoryArray(scip, &((*consdata)->andnegs), andnegs, nandconss) );
1299  }
1300  else
1301  {
1302  SCIP_CALL( SCIPallocClearBlockMemoryArray(scip, &((*consdata)->andnegs), nandconss) );
1303  }
1304  (*consdata)->nconsanddatas = nandconss;
1305  (*consdata)->sconsanddatas = nandconss;
1306 
1307  /* allocate temporary memory */
1308  SCIP_CALL( SCIPallocBufferArray(scip, &andress, nandconss) );
1309 
1310  conshdlrdata = SCIPconshdlrGetData(conshdlr);
1311  assert(conshdlrdata != NULL);
1312  assert(conshdlrdata->hashmap != NULL);
1313 
1314  /* get all and-resultants for sorting */
1315  for( c = nandconss - 1; c >= 0; --c )
1316  {
1317  assert(andconss[c] != NULL);
1318 
1319  andress[c] = SCIPgetResultantAnd(scip, andconss[c]);
1320  assert(andress[c] != NULL);
1321 
1322  (*consdata)->consanddatas[c] = (CONSANDDATA*) SCIPhashmapGetImage(conshdlrdata->hashmap, (void*)andress[c]);
1323  assert((*consdata)->consanddatas[c] != NULL);
1324  assert((*consdata)->consanddatas[c]->origcons == andconss[c] || (*consdata)->consanddatas[c]->cons == andconss[c]);
1325 
1326  if( transforming )
1327  {
1328  /* if we perform a new transformation, we need to capture the transformed constraint */
1329  if( (*consdata)->consanddatas[c]->origcons != NULL && (*consdata)->consanddatas[c]->cons == NULL )
1330  {
1331  SCIP_VAR** vars;
1332  int ncvars;
1333  int v;
1334 
1335  /* do not capture the and constraint when scip is in transformed mode; this automatically happens in
1336  * SCIPtransformCons()
1337  */
1338  SCIP_CALL( SCIPtransformCons(scip, (*consdata)->consanddatas[c]->origcons, &((*consdata)->consanddatas[c]->cons)) );
1339  assert((*consdata)->consanddatas[c]->cons != NULL);
1340  assert((*consdata)->consanddatas[c]->newvars == NULL);
1341  assert((*consdata)->consanddatas[c]->isoriginal);
1342 
1343  (*consdata)->consanddatas[c]->istransformed = TRUE;
1344 
1345  vars = (*consdata)->consanddatas[c]->vars;
1346  ncvars = (*consdata)->consanddatas[c]->nvars;
1347  assert(vars != NULL || ncvars == 0);
1348 
1349  /* get transformed variables */
1350  SCIP_CALL( SCIPgetTransformedVars(scip, ncvars, vars, vars) );
1351 
1352  /* resort variables in transformed problem, because the order might change while tranforming */
1353  SCIPsortPtr((void**)vars, SCIPvarComp, ncvars);
1354 
1355  /* capture all transformed variables */
1356  for( v = ncvars - 1; v >= 0; --v )
1357  {
1358  SCIP_CALL( SCIPcaptureVar(scip, vars[v]) ); /*lint !e613*/
1359  }
1360  }
1361  else if( (*consdata)->consanddatas[c]->cons != NULL )
1362  assert((*consdata)->consanddatas[c]->istransformed);
1363 
1364  ++((*consdata)->consanddatas[c]->nuses);
1365  }
1366  else if( transformed )
1367  {
1368  assert((*consdata)->consanddatas[c]->cons == andconss[c]);
1369  assert(SCIPconsIsTransformed(andconss[c]));
1370  assert((*consdata)->consanddatas[c]->istransformed);
1371  }
1372  }
1373 
1374  /* sort and-constraints after indices of corresponding and-resultants */
1375  SCIPsortPtrPtrRealBool((void**)andress, (void**)((*consdata)->consanddatas), (*consdata)->andcoefs, (*consdata)->andnegs, SCIPvarComp, nandconss);
1376 
1377  /* free temporary memory */
1378  SCIPfreeBufferArray(scip, &andress);
1379  }
1380  else
1381  {
1382  (*consdata)->consanddatas = NULL;
1383  (*consdata)->andcoefs = NULL;
1384  (*consdata)->andnegs = NULL;
1385  (*consdata)->nconsanddatas = 0;
1386  (*consdata)->sconsanddatas = 0;
1387  }
1388 
1389  /* copy left and right hand side */
1390  (*consdata)->lhs = lhs;
1391  (*consdata)->rhs = rhs;
1392 
1393  (*consdata)->changed = TRUE;
1394  (*consdata)->propagated = FALSE;
1395  (*consdata)->presolved = FALSE;
1396  (*consdata)->cliquesadded = FALSE;
1397  (*consdata)->upgradetried = TRUE;
1398 
1399  /* count number of used consanddata objects in original problem */
1400  if( SCIPgetStage(scip) == SCIP_STAGE_PROBLEM )
1401  {
1402  SCIP_CONSHDLRDATA* conshdlrdata;
1403  conshdlrdata = SCIPconshdlrGetData(conshdlr);
1404  assert(conshdlrdata != NULL);
1405 
1406  conshdlrdata->noriguses += (*consdata)->nconsanddatas;
1407  }
1408 
1409  return SCIP_OKAY;
1410 }
1411 
1412 /** free a pseudo boolean constraint data */
1413 static
1415  SCIP*const scip, /**< SCIP data structure */
1416  SCIP_CONSDATA** consdata, /**< pointer to linear constraint data */
1417  SCIP_Bool isorig, /**< are we freeing an original constraint? */
1418  SCIP_CONSHDLRDATA* conshdlrdata /**< constraint handler data */
1419  )
1420 {
1421  CONSANDDATA** consanddatas;
1422  int nconsanddatas;
1423  int c;
1424 
1425  assert(scip != NULL);
1426  assert(consdata != NULL);
1427  assert(*consdata != NULL);
1428  assert((*consdata)->nconsanddatas == 0 || (*consdata)->consanddatas != NULL);
1429  assert(conshdlrdata != NULL);
1430 
1431  /* release linear constraint */
1432  if( (*consdata)->lincons != NULL )
1433  {
1434  SCIP_CALL( SCIPreleaseCons(scip, &((*consdata)->lincons)) );
1435  }
1436 
1437  nconsanddatas = (*consdata)->nconsanddatas;
1438  consanddatas = (*consdata)->consanddatas;
1439 
1440  /* count down uses and if necessary release constraints and delete data from hashtable and -map */
1441  for( c = nconsanddatas - 1; c >= 0; --c )
1442  {
1443  assert((consanddatas[c]->origcons == NULL) == (consanddatas[c]->noriguses == 0));
1444  assert((consanddatas[c]->cons == NULL) == (consanddatas[c]->nuses == 0));
1445  assert(consanddatas[c]->nuses >= 0);
1446  assert(consanddatas[c]->noriguses >= 0);
1447  assert(isorig ? consanddatas[c]->cons == NULL : TRUE);
1448 
1449  /* are we deleteing a transformed constraint */
1450  if( !isorig && consanddatas[c]->cons != NULL )
1451  {
1452  assert(!SCIPconsIsOriginal(consanddatas[c]->cons));
1453 
1454  --(consanddatas[c]->nuses);
1455 
1456  /* if the consanddata is not used anymore, release the constraint and clear the hashmap- and table */
1457  if( consanddatas[c]->nuses == 0 )
1458  {
1459  if( conshdlrdata->inithashmapandtable )
1460  {
1461  assert(conshdlrdata->hashmap != NULL);
1462  assert(conshdlrdata->hashtable != NULL);
1463 
1464  /* remove consanddata from hashtable, if it existed only in transformed space */
1465  if( consanddatas[c]->origcons == NULL )
1466  {
1467  assert(SCIPhashtableExists(conshdlrdata->hashtable, (void*)consanddatas[c]));
1468  SCIP_CALL( SCIPhashtableRemove(conshdlrdata->hashtable, (void*)consanddatas[c]) );
1469  }
1470  assert(SCIPhashmapExists(conshdlrdata->hashmap, (void*)SCIPgetResultantAnd(scip, consanddatas[c]->cons)));
1471  SCIP_CALL( SCIPhashmapRemove(conshdlrdata->hashmap, (void*)SCIPgetResultantAnd(scip, consanddatas[c]->cons)) );
1472  }
1473 
1474  SCIP_CALL( SCIPreleaseCons(scip, &(consanddatas[c]->cons)) );
1475 
1476  /* if the consanddata object was only used in transformed space, delete the memory block */
1477  if( consanddatas[c]->origcons == NULL )
1478  {
1479  int d;
1480 
1481  assert(conshdlrdata->nallconsanddatas > 0);
1482 
1483  for( d = conshdlrdata->nallconsanddatas - 1; d >= 0; --d )
1484  {
1485  if( conshdlrdata->allconsanddatas[d] == consanddatas[c] )
1486  {
1487  --conshdlrdata->nallconsanddatas;
1488 
1489  SCIPfreeBlockMemory(scip, &(conshdlrdata->allconsanddatas[d])); /*lint !e866*/
1490 
1491  conshdlrdata->allconsanddatas[d] = conshdlrdata->allconsanddatas[conshdlrdata->nallconsanddatas];
1492  break;
1493  }
1494  }
1495  assert(d >= 0);
1496  continue;
1497  }
1498  }
1499  }
1500  /* are we deleteing an original constraint */
1501  else if( isorig && consanddatas[c]->origcons != NULL )
1502  {
1503  assert(SCIPconsIsOriginal(consanddatas[c]->origcons));
1504  assert(consanddatas[c]->nuses == 0);
1505  assert(consanddatas[c]->nnewvars == 0);
1506  assert(consanddatas[c]->snewvars == 0);
1507  assert(consanddatas[c]->newvars == NULL);
1508 
1509  --(consanddatas[c]->noriguses);
1510 
1511  /* if the consanddata is not used anymore, release the constraint and clear the hashmap- and table */
1512  if( consanddatas[c]->noriguses == 0 )
1513  {
1514  int d;
1515 
1516  if( conshdlrdata->inithashmapandtable )
1517  {
1518  assert(conshdlrdata->hashmap != NULL);
1519  assert(conshdlrdata->hashtable != NULL);
1520 
1521  assert(SCIPhashtableExists(conshdlrdata->hashtable, (void*)consanddatas[c]));
1522  SCIP_CALL( SCIPhashtableRemove(conshdlrdata->hashtable, (void*)consanddatas[c]) );
1523 
1524  assert(SCIPhashmapExists(conshdlrdata->hashmap, (void*)SCIPgetResultantAnd(scip, consanddatas[c]->origcons)));
1525  SCIP_CALL( SCIPhashmapRemove(conshdlrdata->hashmap, (void*)SCIPgetResultantAnd(scip, consanddatas[c]->origcons)) );
1526  }
1527 
1528  if( consanddatas[c]->vars != NULL )
1529  {
1530  assert(consanddatas[c]->nvars > 0);
1531  assert(consanddatas[c]->svars > 0);
1532  assert(consanddatas[c]->svars >= consanddatas[c]->nvars);
1533 
1534  SCIPfreeBlockMemoryArrayNull(scip, &(consanddatas[c]->vars), consanddatas[c]->svars);
1535  consanddatas[c]->nvars = 0;
1536  consanddatas[c]->svars = 0;
1537  }
1538  else
1539  {
1540  assert(consanddatas[c]->nvars == 0);
1541  assert(consanddatas[c]->svars == 0);
1542  }
1543 
1544  SCIP_CALL( SCIPreleaseCons(scip, &(consanddatas[c]->origcons)) );
1545  assert(consanddatas[c]->origcons == NULL);
1546 
1547  /* delete consanddata object */
1548  assert(conshdlrdata->nallconsanddatas > 0);
1549  for( d = conshdlrdata->nallconsanddatas - 1; d >= 0; --d )
1550  {
1551  if( conshdlrdata->allconsanddatas[d] == consanddatas[c] )
1552  {
1553  --conshdlrdata->nallconsanddatas;
1554 
1555  SCIPfreeBlockMemory(scip, &(conshdlrdata->allconsanddatas[d])); /*lint !e866*/
1556 
1557  conshdlrdata->allconsanddatas[d] = conshdlrdata->allconsanddatas[conshdlrdata->nallconsanddatas];
1558  break;
1559  }
1560  }
1561  assert(d >= 0);
1562 
1563  continue;
1564  }
1565  }
1566  else
1567  {
1568  assert(!consanddatas[c]->istransformed);
1569  assert(consanddatas[c]->cons == NULL);
1570  }
1571 
1572  /* clear and remove capture of transformed consanddata */
1573  if( consanddatas[c]->nuses == 0 && consanddatas[c]->istransformed )
1574  {
1575  SCIP_CALL( transformToOrig(scip, consanddatas[c], conshdlrdata) );
1576  }
1577 #ifndef NDEBUG
1578  else if( consanddatas[c]->nuses == 0 )
1579  {
1580  SCIP_VAR** tmpvars;
1581  int ntmpvars;
1582  int v;
1583 
1584  assert(consanddatas[c]->nnewvars == 0);
1585  assert(consanddatas[c]->snewvars == 0);
1586  assert(consanddatas[c]->newvars == NULL);
1587 
1588  tmpvars = consanddatas[c]->vars;
1589  ntmpvars = consanddatas[c]->nvars;
1590 
1591  /* release all variables */
1592  for( v = ntmpvars - 1; v >= 0; --v )
1593  {
1594  assert(tmpvars[v] != NULL);
1595  assert(SCIPvarIsOriginal(tmpvars[v]));
1596  }
1597  }
1598 #endif
1599 
1600  /* restore original data */
1601  if( !consanddatas[c]->istransformed && consanddatas[c]->noriguses > 0 )
1602  {
1603  assert(consanddatas[c]->origcons != NULL);
1604  assert(consanddatas[c]->nuses == 0);
1605  assert(consanddatas[c]->nnewvars == 0);
1606  assert(consanddatas[c]->snewvars == 0);
1607  assert(consanddatas[c]->newvars == NULL);
1608  assert(consanddatas[c]->nvars > 0);
1609  assert(consanddatas[c]->svars > 0);
1610  assert(consanddatas[c]->svars >= consanddatas[c]->nvars);
1611  assert(consanddatas[c]->vars != NULL);
1612  assert(consanddatas[c]->isoriginal);
1613 
1614  assert(consanddatas[c]->nvars == SCIPgetNVarsAnd(scip, consanddatas[c]->origcons));
1615  assert(SCIPgetVarsAnd(scip, consanddatas[c]->origcons) != NULL);
1616 
1617  /* check that the hash map and tabkle are still having all information */
1618  if( conshdlrdata->inithashmapandtable )
1619  {
1620  assert(conshdlrdata->hashmap != NULL);
1621  assert(conshdlrdata->hashtable != NULL);
1622  assert(SCIPgetResultantAnd(scip, consanddatas[c]->origcons) != NULL);
1623  assert(SCIPhashtableExists(conshdlrdata->hashtable, (void*)consanddatas[c]));
1624  assert(consanddatas[c] == (CONSANDDATA*)(SCIPhashtableRetrieve(conshdlrdata->hashtable, (void*)consanddatas[c])));
1625  assert(SCIPhashmapExists(conshdlrdata->hashmap, (void*)SCIPgetResultantAnd(scip, consanddatas[c]->origcons)));
1626  assert(consanddatas[c] == (CONSANDDATA*)(SCIPhashmapGetImage(conshdlrdata->hashmap, (void*)SCIPgetResultantAnd(scip, consanddatas[c]->origcons))));
1627  }
1628  }
1629  }
1630 
1631  /* free array of and-constraints */
1632  SCIPfreeBlockMemoryArrayNull(scip, &((*consdata)->andnegs), (*consdata)->sconsanddatas);
1633  SCIPfreeBlockMemoryArrayNull(scip, &((*consdata)->andcoefs), (*consdata)->sconsanddatas);
1634  SCIPfreeBlockMemoryArrayNull(scip, &((*consdata)->consanddatas), (*consdata)->sconsanddatas);
1635 
1636  SCIPfreeBlockMemory(scip, consdata);
1637 
1638  return SCIP_OKAY;
1639 }
1640 
1641 /** check the locks of an AND resultant and removes it from all global structures if the resultant is not locked anymore */
1642 static
1644  SCIP*const scip, /**< SCIP data structure */
1645  SCIP_VAR* res /**< resultant of AND constraint */
1646  )
1647 {
1648  assert(scip != NULL);
1649  assert(res != NULL);
1650 
1651  /* the resultant has no locks left and might be dual fixed now, we need to delete all its cliques */
1654  {
1656  }
1657 
1658  return SCIP_OKAY;
1659 }
1660 
1661 /** installs rounding locks for the given and-constraint associated with given coefficient */
1662 static
1664  SCIP*const scip, /**< SCIP data structure */
1665  SCIP_CONS*const cons, /**< pseudoboolean constraint */
1666  CONSANDDATA*const consanddata, /**< CONSANDDATA object for which we want to add the locks */
1667  SCIP_Real const coef, /**< coefficient which led to old locks */
1668  SCIP_Real const lhs, /**< left hand side */
1669  SCIP_Real const rhs /**< right hand side */
1670  )
1671 {
1672  SCIP_VAR** vars;
1673  int nvars;
1674  SCIP_VAR* res;
1675  SCIP_Bool haslhs;
1676  SCIP_Bool hasrhs;
1677  int v;
1678 
1679  assert(scip != NULL);
1680  assert(cons != NULL);
1682  assert(consanddata != NULL);
1683  assert(!SCIPisInfinity(scip, coef) && !SCIPisInfinity(scip, -coef));
1684  assert(!SCIPisInfinity(scip, lhs));
1685  assert(!SCIPisInfinity(scip, -rhs));
1686  assert(SCIPisLE(scip, lhs, rhs));
1687 
1688  /* choose correct variable array to add locks for, we only add locks for now valid variables */
1689  if( consanddata->nnewvars > 0 )
1690  {
1691  vars = consanddata->newvars;
1692  nvars = consanddata->nnewvars;
1693  }
1694  else
1695  {
1696  vars = consanddata->vars;
1697  nvars = consanddata->nvars;
1698  }
1699 
1700  res = SCIPgetResultantAnd(scip, consanddata->cons);
1701  assert(nvars == 0 || (vars != NULL && res != NULL));
1702 
1703  /* check which sites are infinity */
1704  haslhs = !SCIPisInfinity(scip, -lhs);
1705  hasrhs = !SCIPisInfinity(scip, rhs);
1706 
1707  if( SCIPconsIsLocked(cons) )
1708  {
1709  /* locking variables */
1710  if( SCIPisPositive(scip, coef) )
1711  {
1712  for( v = nvars - 1; v >= 0; --v )
1713  {
1714  SCIP_CALL( SCIPlockVarCons(scip, vars[v], cons, haslhs, hasrhs) );
1715  }
1716  }
1717  else
1718  {
1719  for( v = nvars - 1; v >= 0; --v )
1720  {
1721  SCIP_CALL( SCIPlockVarCons(scip, vars[v], cons, hasrhs, haslhs) );
1722  }
1723  }
1724  SCIP_CALL( SCIPlockVarCons(scip, res, cons, TRUE, TRUE) );
1725  }
1726 
1727  return SCIP_OKAY;
1728 }
1729 
1730 /** removes rounding locks for the given and-constraint associated with given coefficient */
1731 static
1733  SCIP*const scip, /**< SCIP data structure */
1734  SCIP_CONS*const cons, /**< pseudoboolean constraint */
1735  CONSANDDATA*const consanddata, /**< CONSANDDATA object for which we want to delete the locks */
1736  SCIP_Real const coef, /**< coefficient which led to old locks */
1737  SCIP_Real const lhs, /**< left hand side which led to old locks */
1738  SCIP_Real const rhs /**< right hand side which led to old locks */
1739  )
1740 {
1741  SCIP_VAR** vars;
1742  int nvars;
1743  SCIP_VAR* res;
1744  SCIP_Bool haslhs;
1745  SCIP_Bool hasrhs;
1746  int v;
1747 
1748  assert(scip != NULL);
1749  assert(cons != NULL);
1751  assert(consanddata != NULL);
1752  assert(!SCIPisInfinity(scip, coef) && !SCIPisInfinity(scip, -coef));
1753  assert(!SCIPisInfinity(scip, lhs));
1754  assert(!SCIPisInfinity(scip, -rhs));
1755  assert(SCIPisLE(scip, lhs, rhs));
1756 
1757  vars = consanddata->vars;
1758  nvars = consanddata->nvars;
1759 
1760  if( consanddata->cons != NULL )
1761  res = SCIPgetResultantAnd(scip, consanddata->cons);
1762  else
1763  res = NULL;
1764  assert(nvars == 0 || vars != NULL);
1765 
1766  /* check which sites are infinity */
1767  haslhs = !SCIPisInfinity(scip, -lhs);
1768  hasrhs = !SCIPisInfinity(scip, rhs);
1769 
1770  if( SCIPconsIsLocked(cons) )
1771  {
1772  /* unlock variables */
1773  if( SCIPisPositive(scip, coef) )
1774  {
1775  for( v = nvars - 1; v >= 0; --v )
1776  {
1777  SCIP_CALL( SCIPunlockVarCons(scip, vars[v], cons, haslhs, hasrhs) );
1778  }
1779  }
1780  else
1781  {
1782  for( v = nvars - 1; v >= 0; --v )
1783  {
1784  SCIP_CALL( SCIPunlockVarCons(scip, vars[v], cons, hasrhs, haslhs) );
1785  }
1786  }
1787 
1788  if( res != NULL )
1789  {
1790  SCIP_CALL( SCIPunlockVarCons(scip, res, cons, TRUE, TRUE) );
1791 
1792  SCIP_CALL( checkLocksAndRes(scip, res) );
1793  }
1794  }
1795 
1796  return SCIP_OKAY;
1797 }
1798 
1799 /** prints pseudoboolean constraint in CIP format to file stream */
1800 static
1802  SCIP*const scip, /**< SCIP data structure */
1803  SCIP_CONS*const cons, /**< pseudoboolean constraint */
1804  FILE*const file /**< output file (or NULL for standard output) */
1805  )
1806 {
1807  SCIP_CONSHDLR* conshdlr;
1808  SCIP_CONSHDLRDATA* conshdlrdata;
1809  SCIP_CONSDATA* consdata;
1810 
1811  SCIP_VAR** vars;
1812  SCIP_Real* coefs;
1813  int nvars;
1814  SCIP_Real lhs;
1815  SCIP_Real rhs;
1816 
1817  SCIP_VAR** linvars;
1818  SCIP_Real* lincoefs;
1819  int nlinvars;
1820  int v;
1821 
1822  SCIP_VAR** andress;
1823  SCIP_Real* andcoefs;
1824  SCIP_Bool* andnegs;
1825  int nandress;
1826 
1827  SCIP_Bool printed;
1828 
1829  assert(scip != NULL);
1830  assert(cons != NULL);
1831 
1832 #ifdef WITHEQKNAPSACK
1833  if( SCIPconsIsDeleted(cons) )
1834  return SCIP_OKAY;
1835 #endif
1836 
1837  consdata = SCIPconsGetData(cons);
1838  assert(consdata != NULL);
1839  assert(consdata->lincons != NULL);
1840  /* more than one and-constraint is needed, otherwise this pseudoboolean constraint should be upgraded to a linear constraint */
1841  assert(consdata->nconsanddatas >= 0);
1842 
1843  /* gets number of variables in linear constraint */
1844  SCIP_CALL( getLinearConsNVars(scip, consdata->lincons, consdata->linconstype, &nvars) );
1845 
1846  /* allocate temporary memory */
1847  SCIP_CALL( SCIPallocBufferArray(scip, &vars, nvars) );
1848  SCIP_CALL( SCIPallocBufferArray(scip, &coefs, nvars) );
1849  SCIP_CALL( SCIPallocBufferArray(scip, &linvars, nvars) );
1850  SCIP_CALL( SCIPallocBufferArray(scip, &lincoefs, nvars) );
1851  SCIP_CALL( SCIPallocBufferArray(scip, &andress, nvars) );
1852  SCIP_CALL( SCIPallocBufferArray(scip, &andcoefs, nvars) );
1853  SCIP_CALL( SCIPallocBufferArray(scip, &andnegs, nvars) );
1854 
1855  /* get sides of linear constraint */
1856  SCIP_CALL( getLinearConsSides(scip, consdata->lincons, consdata->linconstype, &lhs, &rhs) );
1857  assert(!SCIPisInfinity(scip, lhs));
1858  assert(!SCIPisInfinity(scip, -rhs));
1859  assert(SCIPisLE(scip, lhs, rhs));
1860 
1861  /* get variables and coefficient of linear constraint */
1862  SCIP_CALL( getLinearConsVarsData(scip, consdata->lincons, consdata->linconstype, vars, coefs, &nvars) );
1863  assert(nvars == 0 || (coefs != NULL));
1864 
1865  /* calculate all not artificial linear variables and all artificial and-resultants which will be ordered like the
1866  * 'consanddatas' such that the and-resultant of the and-constraint is the and-resultant in the 'andress' array
1867  * afterwards
1868  */
1869  SCIP_CALL( getLinVarsAndAndRess(scip, cons, vars, coefs, nvars, linvars, lincoefs, &nlinvars,
1870  andress, andcoefs, andnegs, &nandress) );
1871  assert(consdata->nconsanddatas == nandress);
1872 
1873  /* number of variables should be consistent, number of 'real' linear variables plus number of and-constraints should
1874  * have to be equal to the number of variables in the linear constraint
1875  */
1876  assert(consdata->nlinvars + consdata->nconsanddatas == nvars);
1877 
1878  /* print left hand side for ranged rows */
1879  if( !SCIPisInfinity(scip, -lhs) && !SCIPisInfinity(scip, rhs) && !SCIPisEQ(scip, lhs, rhs) )
1880  SCIPinfoMessage(scip, file, "%.15g <= ", lhs);
1881 
1882  printed = FALSE;
1883 
1884  /* print coefficients and variables */
1885  if( nlinvars > 0)
1886  {
1887  printed= TRUE;
1888 
1889  /* print linear part of constraint */
1890  SCIP_CALL( SCIPwriteVarsLinearsum(scip, file, linvars, lincoefs, nlinvars, TRUE) );
1891  }
1892 
1893  conshdlr = SCIPconsGetHdlr(cons);
1894  assert(conshdlr != NULL);
1895  conshdlrdata = SCIPconshdlrGetData(conshdlr);
1896  assert(conshdlrdata != NULL);
1897  assert(conshdlrdata->hashmap != NULL);
1898 
1899  /* print all non-linear terms */
1900  for( v = nandress - 1; v >= 0; --v )
1901  {
1902  CONSANDDATA* consanddata;
1903  SCIP_CONS* andcons;
1904  SCIP_VAR** andvars;
1905  int nandvars;
1906 
1907  if( !SCIPconsIsOriginal(cons) )
1908  {
1909  /* if the and resultant was fixed we print a constant */
1910  if( SCIPvarGetLbLocal(andress[v]) > 0.5 || SCIPvarGetUbLocal(andress[v]) < 0.5 )
1911  {
1912  if( SCIPvarGetLbGlobal(andress[v]) > 0.5 )
1913  {
1914  printed = TRUE;
1915  SCIPinfoMessage(scip, file, " %+.15g ", andcoefs[v] * SCIPvarGetLbGlobal(andress[v]));
1916  }
1917  continue;
1918  }
1919  else if( SCIPvarGetStatus(andress[v]) == SCIP_VARSTATUS_AGGREGATED )
1920  {
1921  SCIP_VAR* aggrvar;
1922  SCIP_Bool negated;
1923 
1924  SCIP_CALL( SCIPgetBinvarRepresentative(scip, andress[v], &aggrvar, &negated) );
1925  assert(aggrvar != NULL);
1926  assert(SCIPvarGetType(aggrvar) == SCIP_VARTYPE_BINARY);
1927 
1928  printed = TRUE;
1929  SCIPinfoMessage(scip, file, " %+.15g %s<%s>[B]", andcoefs[v], negated ? "~" : "", SCIPvarGetName(aggrvar));
1930 
1931  continue;
1932  }
1933  }
1934 
1935  consanddata = (CONSANDDATA*) SCIPhashmapGetImage(conshdlrdata->hashmap, (void*)andress[v]);
1936  assert(consanddata != NULL);
1937 
1938  if( SCIPconsIsOriginal(cons) )
1939  andcons = consanddata->origcons;
1940  else
1941  andcons = consanddata->cons;
1942  assert(andcons != NULL);
1943 
1944  andvars = SCIPgetVarsAnd(scip, andcons);
1945  nandvars = SCIPgetNVarsAnd(scip, andcons);
1946  assert(nandvars == 0 || andvars != NULL);
1947 
1948  if( nandvars > 0 )
1949  {
1950  printed = TRUE;
1951  SCIPinfoMessage(scip, file, " %+.15g %s(", andcoefs[v], andnegs[v] ? "~" : "");
1952 
1953  /* @todo: better write new method SCIPwriteProduct */
1954  /* print variable list */
1955  SCIP_CALL( SCIPwriteVarsList(scip, file, andvars, nandvars, TRUE, '*') );
1956 
1957  SCIPinfoMessage(scip, file, ")");
1958  }
1959  }
1960 
1961  if( !printed )
1962  {
1963  SCIPinfoMessage(scip, file, " 0 ");
1964  }
1965 
1966  /* free temporary memory */
1967  SCIPfreeBufferArray(scip, &andnegs);
1968  SCIPfreeBufferArray(scip, &andcoefs);
1969  SCIPfreeBufferArray(scip, &andress);
1970  SCIPfreeBufferArray(scip, &lincoefs);
1971  SCIPfreeBufferArray(scip, &linvars);
1972  SCIPfreeBufferArray(scip, &coefs);
1973  SCIPfreeBufferArray(scip, &vars);
1974 
1975  /* print right hand side */
1976  if( SCIPisEQ(scip, lhs, rhs) )
1977  SCIPinfoMessage(scip, file, "== %.15g", rhs);
1978  else if( !SCIPisInfinity(scip, rhs) )
1979  SCIPinfoMessage(scip, file, "<= %.15g", rhs);
1980  else if( !SCIPisInfinity(scip, -lhs) )
1981  SCIPinfoMessage(scip, file, ">= %.15g", lhs);
1982  else
1983  SCIPinfoMessage(scip, file, " [free]");
1984 
1985  return SCIP_OKAY;
1986 }
1987 
1988 /** creates and/or adds the resultant for a given term */
1989 static
1991  SCIP*const scip, /**< SCIP data structure */
1992  SCIP_CONSHDLR*const conshdlr, /**< pseudoboolean constraint handler */
1993  SCIP_VAR**const vars, /**< array of variables to get and-constraints for */
1994  int const nvars, /**< number of variables to get and-constraints for */
1995  SCIP_Bool const initial, /**< should the LP relaxation of constraint be in the initial LP?
1996  * Usually set to TRUE. Set to FALSE for 'lazy constraints'. */
1997  SCIP_Bool const enforce, /**< should the constraint be enforced during node processing?
1998  * TRUE for model constraints, FALSE for additional, redundant
1999  * constraints. */
2000  SCIP_Bool const check, /**< should the constraint be checked for feasibility?
2001  * TRUE for model constraints, FALSE for additional, redundant
2002  * constraints. */
2003  SCIP_Bool const local, /**< is constraint only valid locally?
2004  * Usually set to FALSE. Has to be set to TRUE, e.g., for branching
2005  * constraints. */
2006  SCIP_Bool const modifiable, /**< is constraint modifiable (subject to column generation)?
2007  * Usually set to FALSE. In column generation applications, set to TRUE
2008  * if pricing adds coefficients to this constraint. */
2009  SCIP_Bool const dynamic, /**< is constraint subject to aging?
2010  * Usually set to FALSE. Set to TRUE for own cuts which
2011  * are seperated as constraints. */
2012  SCIP_Bool const stickingatnode, /**< should the constraint always be kept at the node where it was added, even
2013  * if it may be moved to a more global node?
2014  * Usually set to FALSE. Set to TRUE to for constraints that represent
2015  * node data. */
2016  SCIP_CONS**const andcons /**< pointer to store and-constraint */
2017  )
2018 {
2019  CONSANDDATA* newdata;
2020  CONSANDDATA* tmpdata;
2021  SCIP_CONSHDLRDATA* conshdlrdata;
2022  char name[SCIP_MAXSTRLEN];
2023  SCIP_Bool separate;
2024  SCIP_Bool propagate;
2025  SCIP_Bool removable;
2026  SCIP_Bool transformed;
2027 
2028  assert(scip != NULL);
2029  assert(conshdlr != NULL);
2030  assert(vars != NULL);
2031  assert(nvars > 0);
2032  assert(andcons != NULL);
2033 
2034  conshdlrdata = SCIPconshdlrGetData(conshdlr);
2035  assert(conshdlrdata != NULL);
2036  assert(conshdlrdata->hashtable != NULL);
2037 
2038  transformed = SCIPisTransformed(scip);
2039 
2040  /* allocate memory for a possible new consanddata object */
2041  SCIP_CALL( SCIPallocBlockMemory(scip, &newdata) );
2042  SCIP_CALL( SCIPduplicateBlockMemoryArray(scip, &(newdata->vars), vars, nvars) );
2043  newdata->nvars = nvars;
2044  newdata->svars = nvars;
2045  newdata->newvars = NULL;
2046  newdata->nnewvars = 0;
2047  newdata->snewvars = 0;
2048  newdata->noriguses = 0;
2049  newdata->nuses = 0;
2050  newdata->istransformed = transformed;
2051  newdata->isoriginal = !transformed;
2052  newdata->cons = NULL;
2053  newdata->origcons = NULL;
2054 
2055  /* sort variables */
2056  SCIPsortPtr((void**)(newdata->vars), SCIPvarComp, nvars);
2057 
2058  /* get constraint from current hash table with same variables as cons0 */
2059  tmpdata = (CONSANDDATA*)(SCIPhashtableRetrieve(conshdlrdata->hashtable, (void*)newdata));
2060 
2061  /* if there is already the same and constraint created use this resultant */
2062  if( tmpdata != NULL )
2063  {
2064 #ifndef NDEBUG
2065  SCIP_VAR* res;
2066 #endif
2067  if( transformed )
2068  {
2069  assert(tmpdata->cons != NULL);
2070  *andcons = tmpdata->cons;
2071 
2072  assert(tmpdata->nuses > 0);
2073  /* increase usage of data object */
2074  ++(tmpdata->nuses);
2075  }
2076  else
2077  {
2078  assert(tmpdata->origcons != NULL);
2079  *andcons = tmpdata->origcons;
2080 
2081  assert(tmpdata->noriguses > 0);
2082  /* increase usage of data object */
2083  ++(tmpdata->noriguses);
2084  }
2085  assert(*andcons != NULL);
2086 
2087 #ifndef NDEBUG
2088  res = SCIPgetResultantAnd(scip, *andcons);
2089  assert(res != NULL);
2090 
2091  /* check that we already have added this resultant to and-constraint entry */
2092  assert(SCIPhashmapExists(conshdlrdata->hashmap, (void*)res));
2093 #endif
2094  }
2095  else
2096  {
2097  /* create new and-constraint */
2098  SCIP_CONS* newcons;
2099  SCIP_VAR* resultant;
2100 
2101  /* create auxiliary variable */
2102  (void)SCIPsnprintf(name, SCIP_MAXSTRLEN, ARTIFICIALVARNAMEPREFIX"%d", conshdlrdata->nallconsanddatas);
2103  SCIP_CALL( SCIPcreateVar(scip, &resultant, name, 0.0, 1.0, 0.0, SCIP_VARTYPE_BINARY,
2104  TRUE, TRUE, NULL, NULL, NULL, NULL, NULL) );
2105 
2106 #if 1 /* @todo: check whether we want to branch on artificial variables, the test results show that it is of advantage */
2107  /* change branching priority of artificial variable to -1 */
2108  SCIP_CALL( SCIPchgVarBranchPriority(scip, resultant, -1) );
2109 #endif
2110 
2111  /* add auxiliary variable to the problem */
2112  SCIP_CALL( SCIPaddVar(scip, resultant) );
2113 
2114 #if 0 /* does not work for since the value of artificial resultants must not be equal to the value computed by their
2115  * product, since these variables are irrelevant */
2116 #ifdef WITH_DEBUG_SOLUTION
2117  if( SCIPdebugIsMainscip(scip) )
2118  {
2119  SCIP_Real val;
2120  SCIP_Real debugsolval;
2121  int v;
2122 
2123  for( v = nvars - 1; v >= 0; --v )
2124  {
2125  SCIP_CALL( SCIPdebugGetSolVal(scip, vars[v], &val) );
2126  assert(SCIPisFeasZero(scip, val) || SCIPisFeasEQ(scip, val, 1.0));
2127 
2128  if( val < 0.5 )
2129  break;
2130  }
2131  val = ((val < 0.5) ? 0.0 : 1.0);
2132 
2133  SCIP_CALL( SCIPdebugGetSolVal(scip, resultant, &debugsolval) );
2134  if( (SCIPvarIsOriginal(resultant) || SCIPvarIsTransformedOrigvar(resultant)) && !SCIPisFeasEQ(scip, debugsolval, val) )
2135  {
2136  SCIPerrorMessage("computed solution value %g for resultant <%s> violates debug solution value %g\n", val, SCIPvarGetName(resultant), debugsolval);
2137  SCIPABORT();
2138  return SCIP_ERROR; /*lint !e527*/
2139  }
2140  else if( !SCIPvarIsOriginal(resultant) && !SCIPvarIsTransformedOrigvar(resultant) )
2141  {
2142  SCIP_CALL( SCIPdebugAddSolVal(scip, resultant, val) );
2143  }
2144  }
2145 #endif
2146 #endif
2147 
2148  SCIP_CALL( SCIPgetBoolParam(scip, "constraints/" CONSHDLR_NAME "/nlcseparate", &separate) );
2149  SCIP_CALL( SCIPgetBoolParam(scip, "constraints/" CONSHDLR_NAME "/nlcpropagate", &propagate) );
2150  SCIP_CALL( SCIPgetBoolParam(scip, "constraints/" CONSHDLR_NAME "/nlcremovable", &removable) );
2151 
2152  /* we do not want to check the and constraints, so the check flag will be FALSE */
2153 
2154  /* create and add "and" constraint for the multiplication of the binary variables */
2155  (void)SCIPsnprintf(name, SCIP_MAXSTRLEN, "andcons_%d", conshdlrdata->nallconsanddatas);
2156  SCIP_CALL( SCIPcreateConsAnd(scip, &newcons, name, resultant, newdata->nvars, newdata->vars,
2157  initial, separate, enforce, check && FALSE, propagate,
2158  local, modifiable, dynamic, removable, stickingatnode) ); /*lint !e506*/
2159  SCIP_CALL( SCIPaddCons(scip, newcons) );
2160  SCIPdebugPrintCons(scip, newcons, NULL);
2161 
2162  /* force all deriving constraint from this and constraint to be checked and not removable */
2163  SCIP_CALL( SCIPchgAndConsCheckFlagWhenUpgr(scip, newcons, TRUE) );
2165 
2166  *andcons = newcons;
2167  assert(*andcons != NULL);
2168 
2169  /* resize data for all and-constraints if necessary */
2170  if( conshdlrdata->nallconsanddatas == conshdlrdata->sallconsanddatas )
2171  {
2172  SCIP_CALL( SCIPensureBlockMemoryArray(scip, &(conshdlrdata->allconsanddatas), &(conshdlrdata->sallconsanddatas), SCIPcalcMemGrowSize(scip, conshdlrdata->sallconsanddatas + 1)) );
2173  }
2174 
2175  /* add new data object to global hash table */
2176  conshdlrdata->allconsanddatas[conshdlrdata->nallconsanddatas] = newdata;
2177  ++(conshdlrdata->nallconsanddatas);
2178 
2179  if( transformed )
2180  {
2181  int v;
2182 
2183  newdata->cons = newcons;
2184  SCIP_CALL( SCIPcaptureCons(scip, newdata->cons) );
2185 
2186  /* initialize usage of data object */
2187  newdata->nuses = 1;
2188 
2189  /* capture all variables */
2190  for( v = newdata->nvars - 1; v >= 0; --v )
2191  {
2192  SCIP_CALL( SCIPcaptureVar(scip, newdata->vars[v]) ); /*lint !e613*/
2193  }
2194  }
2195  else
2196  {
2197  newdata->origcons = newcons;
2198  SCIP_CALL( SCIPcaptureCons(scip, newdata->origcons) );
2199 
2200  /* initialize usage of data object */
2201  newdata->noriguses = 1;
2202  }
2203 
2204  /* no such and-constraint in current hash table: insert the new object into hash table */
2205  SCIP_CALL( SCIPhashtableInsert(conshdlrdata->hashtable, (void*)newdata) );
2206 
2207  /* insert new mapping */
2208  assert(!SCIPhashmapExists(conshdlrdata->hashmap, (void*)resultant));
2209  SCIP_CALL( SCIPhashmapInsert(conshdlrdata->hashmap, (void*)resultant, (void*)newdata) );
2210 
2211  /* release and-resultant and -constraint */
2212  SCIP_CALL( SCIPreleaseVar(scip, &resultant) );
2213  SCIP_CALL( SCIPreleaseCons(scip, &newcons) );
2214 
2215  return SCIP_OKAY;
2216  }
2217 
2218  /* free memory */
2219  SCIPfreeBlockMemoryArray(scip, &(newdata->vars), newdata->svars);
2220  SCIPfreeBlockMemory(scip, &newdata);
2221 
2222  return SCIP_OKAY;
2223 }
2224 
2225 /** adds a term to the given pseudoboolean constraint */
2226 static
2228  SCIP*const scip, /**< SCIP data structure */
2229  SCIP_CONS*const cons, /**< pseudoboolean constraint */
2230  SCIP_VAR**const vars, /**< variables of the nonlinear term */
2231  int const nvars, /**< number of variables of the nonlinear term */
2232  SCIP_Real const val /**< coefficient of constraint entry */
2233  )
2234 {
2235  SCIP_CONSHDLR* conshdlr;
2236  SCIP_CONSHDLRDATA* conshdlrdata;
2237  SCIP_CONS* andcons;
2238  SCIP_CONSDATA* consdata;
2239  SCIP_VAR* res;
2240 
2241  assert(scip != NULL);
2242  assert(cons != NULL);
2243  assert(nvars == 0 || vars != NULL);
2244 
2245  if( nvars == 0 || SCIPisZero(scip, val) )
2246  return SCIP_OKAY;
2247 
2248  consdata = SCIPconsGetData(cons);
2249  assert(consdata != NULL);
2250 
2251  conshdlr = SCIPconsGetHdlr(cons);
2252  assert(conshdlr != NULL);
2253 
2254  conshdlrdata = SCIPconshdlrGetData(conshdlr);
2255  assert(conshdlrdata != NULL);
2256 
2257  /* create (and add) and-constraint */
2258  SCIP_CALL( createAndAddAndCons(scip, conshdlr, vars, nvars,
2261  &andcons) );
2262  assert(andcons != NULL);
2263 
2264  /* ensure memory size */
2265  if( consdata->nconsanddatas == consdata->sconsanddatas )
2266  {
2267  SCIP_CALL( SCIPensureBlockMemoryArray(scip, &(consdata->consanddatas), &(consdata->sconsanddatas), consdata->sconsanddatas + 1) );
2268  }
2269 
2270  res = SCIPgetResultantAnd(scip, andcons);
2271  assert(res != NULL);
2272  assert(SCIPhashmapGetImage(conshdlrdata->hashmap, (void*)res) != NULL);
2273 
2274  consdata->consanddatas[consdata->nconsanddatas] = (CONSANDDATA*) SCIPhashmapGetImage(conshdlrdata->hashmap, (void*)res);
2275  ++(consdata->nconsanddatas);
2276 
2277  /* add auxiliary variables to linear constraint */
2278  switch( consdata->linconstype )
2279  {
2281  SCIP_CALL( SCIPaddCoefLinear(scip, consdata->lincons, res, val) );
2282  break;
2284  if( !SCIPisEQ(scip, val, 1.0) )
2285  return SCIP_INVALIDDATA;
2286 
2287  SCIP_CALL( SCIPaddCoefLogicor(scip, consdata->lincons, res) );
2288  break;
2290  if( !SCIPisIntegral(scip, val) || !SCIPisPositive(scip, val) )
2291  return SCIP_INVALIDDATA;
2292 
2293  SCIP_CALL( SCIPaddCoefKnapsack(scip, consdata->lincons, res, (SCIP_Longint) val) );
2294  break;
2296  if( !SCIPisEQ(scip, val, 1.0) )
2297  return SCIP_INVALIDDATA;
2298 
2299  SCIP_CALL( SCIPaddCoefSetppc(scip, consdata->lincons, res) );
2300  break;
2301 #ifdef WITHEQKNAPSACK
2302  case SCIP_LINEARCONSTYPE_EQKNAPSACK:
2303  if( !SCIPisIntegral(scip, val) || !SCIPisPositive(scip, val) )
2304  return SCIP_INVALIDDATA;
2305 
2306  SCIP_CALL( SCIPaddCoefEQKnapsack(scip, consdata->lincons, res, (SCIP_Longint) val) );
2307  break;
2308 #endif
2310  default:
2311  SCIPerrorMessage("unknown linear constraint type\n");
2312  return SCIP_INVALIDDATA;
2313  }
2314 
2315  /* install rounding locks for all new variable */
2316  SCIP_CALL( lockRoundingAndCons(scip, cons, consdata->consanddatas[consdata->nconsanddatas - 1], val, consdata->lhs, consdata->rhs) );
2317 
2318  /* change flags */
2319  consdata->changed = TRUE;
2320  consdata->propagated = FALSE;
2321  consdata->presolved = FALSE;
2322  consdata->cliquesadded = FALSE;
2323  consdata->upgradetried = FALSE;
2324 
2325  return SCIP_OKAY;
2326 }
2327 
2328 /** changes left hand side of linear constraint */
2329 static
2331  SCIP*const scip, /**< SCIP data structure */
2332  SCIP_CONS*const cons, /**< linear constraint */
2333  SCIP_LINEARCONSTYPE const constype, /**< linear constraint type */
2334  SCIP_Real const lhs /**< new left hand side of linear constraint */
2335  )
2336 {
2337  switch( constype )
2338  {
2340  SCIP_CALL( SCIPchgLhsLinear(scip, cons, lhs) );
2341  break;
2345  SCIPerrorMessage("changing left hand side only allowed on standard lienar constraint \n");
2346  return SCIP_INVALIDDATA;
2347 #ifdef WITHEQKNAPSACK
2348  case SCIP_LINEARCONSTYPE_EQKNAPSACK:
2349 #endif
2351  default:
2352  SCIPerrorMessage("unknown linear constraint type\n");
2353  return SCIP_INVALIDDATA;
2354  }
2355 
2356  return SCIP_OKAY;
2357 }
2358 
2359 /** changes right hand side of linear constraint */
2360 static
2362  SCIP*const scip, /**< SCIP data structure */
2363  SCIP_CONS*const cons, /**< linear constraint */
2364  SCIP_LINEARCONSTYPE const constype, /**< linear constraint type */
2365  SCIP_Real const rhs /**< new right hand side of linear constraint */
2366  )
2367 {
2368  switch( constype )
2369  {
2371  SCIP_CALL( SCIPchgRhsLinear(scip, cons, rhs) );
2372  break;
2376  SCIPerrorMessage("changing left hand side only allowed on standard lienar constraint \n");
2377  return SCIP_INVALIDDATA;
2378 #ifdef WITHEQKNAPSACK
2379  case SCIP_LINEARCONSTYPE_EQKNAPSACK:
2380 #endif
2382  default:
2383  SCIPerrorMessage("unknown linear constraint type\n");
2384  return SCIP_INVALIDDATA;
2385  }
2386 
2387  return SCIP_OKAY;
2388 }
2389 
2390 /** sets left hand side of linear constraint */
2391 static
2393  SCIP*const scip, /**< SCIP data structure */
2394  SCIP_CONS*const cons, /**< linear constraint */
2395  SCIP_Real lhs /**< new left hand side */
2396  )
2397 {
2398  SCIP_CONSDATA* consdata;
2399  SCIP_VAR** vars;
2400  SCIP_Real* coefs;
2401  int nvars;
2402  SCIP_VAR** linvars;
2403  SCIP_Real* lincoefs;
2404  int nlinvars;
2405  SCIP_VAR** andress;
2406  SCIP_Real* andcoefs;
2407  SCIP_Bool* andnegs;
2408  int nandress;
2409  SCIP_Real oldlhs;
2410  SCIP_Real oldrhs;
2411 
2412  assert(scip != NULL);
2413  assert(cons != NULL);
2415  assert(!SCIPisInfinity(scip, lhs));
2416 
2417  /* adjust value to not be smaller than -inf */
2418  if ( SCIPisInfinity(scip, -lhs) )
2419  lhs = -SCIPinfinity(scip);
2420 
2421  consdata = SCIPconsGetData(cons);
2422  assert(consdata != NULL);
2423 
2424  /* get sides of linear constraint */
2425  SCIP_CALL( getLinearConsSides(scip, consdata->lincons, consdata->linconstype, &oldlhs, &oldrhs) );
2426  assert(!SCIPisInfinity(scip, oldlhs));
2427  assert(!SCIPisInfinity(scip, -oldrhs));
2428  assert(SCIPisLE(scip, oldlhs, oldrhs));
2429 
2430  /* check whether the side is not changed */
2431  if( SCIPisEQ(scip, oldlhs, lhs) )
2432  return SCIP_OKAY;
2433 
2434  /* gets number of variables in linear constraint */
2435  SCIP_CALL( getLinearConsNVars(scip, consdata->lincons, consdata->linconstype, &nvars) );
2436 
2437  /* allocate temporary memory */
2438  SCIP_CALL( SCIPallocBufferArray(scip, &vars, nvars) );
2439  SCIP_CALL( SCIPallocBufferArray(scip, &coefs, nvars) );
2440  SCIP_CALL( SCIPallocBufferArray(scip, &linvars, nvars) );
2441  SCIP_CALL( SCIPallocBufferArray(scip, &lincoefs, nvars) );
2442  SCIP_CALL( SCIPallocBufferArray(scip, &andress, nvars) );
2443  SCIP_CALL( SCIPallocBufferArray(scip, &andcoefs, nvars) );
2444  SCIP_CALL( SCIPallocBufferArray(scip, &andnegs, nvars) );
2445 
2446  /* get variables and coefficient of linear constraint */
2447  SCIP_CALL( getLinearConsVarsData(scip, consdata->lincons, consdata->linconstype, vars, coefs, &nvars) );
2448  assert(nvars == 0 || (coefs != NULL));
2449 
2450  /* calculate all not artificial linear variables and all artificial and-resultants which will be ordered like the
2451  * 'consanddatas' such that the and-resultant of the and-constraint is the and-resultant in the 'andress' array
2452  * afterwards
2453  */
2454  SCIP_CALL( getLinVarsAndAndRess(scip, cons, vars, coefs, nvars, linvars, lincoefs, &nlinvars, andress, andcoefs, andnegs, &nandress) );
2455  assert(consdata->nconsanddatas == nandress);
2456 
2457  /* if necessary, update the rounding locks of variables */
2458  if( SCIPconsIsLocked(cons) )
2459  {
2460  SCIP_VAR** andvars;
2461  int nandvars;
2462  SCIP_Real val;
2463  int v;
2464  int c;
2465 
2466  assert(SCIPconsIsTransformed(cons));
2467 
2468  if( SCIPisInfinity(scip, -oldlhs) && !SCIPisInfinity(scip, -lhs) )
2469  {
2470  /* non-linear part */
2471  for( c = consdata->nconsanddatas - 1; c >= 0; --c )
2472  {
2473  CONSANDDATA* consanddata;
2474  SCIP_CONS* andcons;
2475 
2476  consanddata = consdata->consanddatas[c];
2477  assert(consanddata != NULL);
2478 
2479  andcons = consanddata->cons;
2480  assert(andcons != NULL);
2481 
2482  andvars = SCIPgetVarsAnd(scip, andcons);
2483  nandvars = SCIPgetNVarsAnd(scip, andcons);
2484  val = andnegs[c] ? -andcoefs[c] : andcoefs[c];
2485 
2486  /* lock variables */
2487  if( SCIPisPositive(scip, val) )
2488  {
2489  for( v = nandvars - 1; v >= 0; --v )
2490  {
2491  SCIP_CALL( SCIPlockVarCons(scip, andvars[v], cons, TRUE, FALSE) );
2492  }
2493  }
2494  else
2495  {
2496  for( v = nandvars - 1; v >= 0; --v )
2497  {
2498  SCIP_CALL( SCIPlockVarCons(scip, andvars[v], cons, FALSE, TRUE) );
2499  }
2500  }
2501  }
2502  }
2503  else if( !SCIPisInfinity(scip, -oldlhs) && SCIPisInfinity(scip, -lhs) )
2504  {
2505  /* non-linear part */
2506  for( c = consdata->nconsanddatas - 1; c >= 0; --c )
2507  {
2508  CONSANDDATA* consanddata;
2509  SCIP_CONS* andcons;
2510 
2511  consanddata = consdata->consanddatas[c];
2512  assert(consanddata != NULL);
2513 
2514  andcons = consanddata->cons;
2515  assert(andcons != NULL);
2516 
2517  andvars = SCIPgetVarsAnd(scip, andcons);
2518  nandvars = SCIPgetNVarsAnd(scip, andcons);
2519  val = andnegs[c] ? -andcoefs[c] : andcoefs[c];
2520 
2521  /* lock variables */
2522  if( SCIPisPositive(scip, val) )
2523  {
2524  for( v = nandvars - 1; v >= 0; --v )
2525  {
2526  SCIP_CALL( SCIPunlockVarCons(scip, andvars[v], cons, TRUE, FALSE) );
2527  }
2528  }
2529  else
2530  {
2531  for( v = nandvars - 1; v >= 0; --v )
2532  {
2533  SCIP_CALL( SCIPunlockVarCons(scip, andvars[v], cons, FALSE, TRUE) );
2534  }
2535  }
2536  }
2537  }
2538  }
2539 
2540  /* check whether the left hand side is increased, if and only if that's the case we maybe can propagate, tighten and add more cliques */
2541  if( SCIPisLT(scip, oldlhs, lhs) )
2542  {
2543  consdata->propagated = FALSE;
2544  }
2545 
2546  /* set new left hand side and update constraint data */
2547  SCIP_CALL( chgLhsLinearCons(scip, consdata->lincons, consdata->linconstype, lhs) );
2548  consdata->lhs = lhs;
2549  consdata->presolved = FALSE;
2550  consdata->changed = TRUE;
2551 
2552  /* free temporary memory */
2553  SCIPfreeBufferArray(scip, &andnegs);
2554  SCIPfreeBufferArray(scip, &andcoefs);
2555  SCIPfreeBufferArray(scip, &andress);
2556  SCIPfreeBufferArray(scip, &lincoefs);
2557  SCIPfreeBufferArray(scip, &linvars);
2558  SCIPfreeBufferArray(scip, &coefs);
2559  SCIPfreeBufferArray(scip, &vars);
2560 
2561  return SCIP_OKAY;
2562 }
2563 
2564 /** sets right hand side of pseudoboolean constraint */
2565 static
2567  SCIP*const scip, /**< SCIP data structure */
2568  SCIP_CONS*const cons, /**< linear constraint */
2569  SCIP_Real rhs /**< new right hand side */
2570  )
2571 {
2572  SCIP_CONSDATA* consdata;
2573  SCIP_VAR** vars;
2574  SCIP_Real* coefs;
2575  int nvars;
2576  SCIP_VAR** linvars;
2577  SCIP_Real* lincoefs;
2578  int nlinvars;
2579  SCIP_VAR** andress;
2580  SCIP_Real* andcoefs;
2581  SCIP_Bool* andnegs;
2582  int nandress;
2583  SCIP_Real oldlhs;
2584  SCIP_Real oldrhs;
2585 
2586  assert(scip != NULL);
2587  assert(cons != NULL);
2589  assert(!SCIPisInfinity(scip, -rhs));
2590 
2591  /* adjust value to not be larger than inf */
2592  if( SCIPisInfinity(scip, rhs) )
2593  rhs = SCIPinfinity(scip);
2594 
2595  consdata = SCIPconsGetData(cons);
2596  assert(consdata != NULL);
2597 
2598  /* get sides of linear constraint */
2599  SCIP_CALL( getLinearConsSides(scip, consdata->lincons, consdata->linconstype, &oldlhs, &oldrhs) );
2600  assert(!SCIPisInfinity(scip, oldlhs));
2601  assert(!SCIPisInfinity(scip, -oldrhs));
2602  assert(SCIPisLE(scip, oldlhs, oldrhs));
2603 
2604  /* check whether the side is not changed */
2605  if( SCIPisEQ(scip, oldrhs, rhs) )
2606  return SCIP_OKAY;
2607 
2608  /* gets number of variables in linear constraint */
2609  SCIP_CALL( getLinearConsNVars(scip, consdata->lincons, consdata->linconstype, &nvars) );
2610 
2611  /* allocate temporary memory */
2612  SCIP_CALL( SCIPallocBufferArray(scip, &vars, nvars) );
2613  SCIP_CALL( SCIPallocBufferArray(scip, &coefs, nvars) );
2614  SCIP_CALL( SCIPallocBufferArray(scip, &linvars, nvars) );
2615  SCIP_CALL( SCIPallocBufferArray(scip, &lincoefs, nvars) );
2616  SCIP_CALL( SCIPallocBufferArray(scip, &andress, nvars) );
2617  SCIP_CALL( SCIPallocBufferArray(scip, &andcoefs, nvars) );
2618  SCIP_CALL( SCIPallocBufferArray(scip, &andnegs, nvars) );
2619 
2620  /* get variables and coefficient of linear constraint */
2621  SCIP_CALL( getLinearConsVarsData(scip, consdata->lincons, consdata->linconstype, vars, coefs, &nvars) );
2622  assert(nvars == 0 || (coefs != NULL));
2623 
2624  /* calculate all not artificial linear variables and all artificial and-resultants which will be ordered like the
2625  * 'consanddatas' such that the and-resultant of the and-constraint is the and-resultant in the 'andress' array
2626  * afterwards
2627  */
2628  SCIP_CALL( getLinVarsAndAndRess(scip, cons, vars, coefs, nvars, linvars, lincoefs, &nlinvars, andress, andcoefs, andnegs, &nandress) );
2629  assert(consdata->nconsanddatas == nandress);
2630 
2631  /* if necessary, update the rounding locks of variables */
2632  if( SCIPconsIsLocked(cons) )
2633  {
2634  SCIP_VAR** andvars;
2635  int nandvars;
2636  SCIP_Real val;
2637  int v;
2638  int c;
2639 
2640  assert(SCIPconsIsTransformed(cons));
2641 
2642  if( SCIPisInfinity(scip, oldrhs) && !SCIPisInfinity(scip, rhs) )
2643  {
2644  /* non-linear part */
2645  for( c = consdata->nconsanddatas - 1; c >= 0; --c )
2646  {
2647  CONSANDDATA* consanddata;
2648  SCIP_CONS* andcons;
2649 
2650  consanddata = consdata->consanddatas[c];
2651  assert(consanddata != NULL);
2652 
2653  andcons = consanddata->cons;
2654  assert(andcons != NULL);
2655 
2656  andvars = SCIPgetVarsAnd(scip, andcons);
2657  nandvars = SCIPgetNVarsAnd(scip, andcons);
2658  val = andnegs[c] ? -andcoefs[c] : andcoefs[c];
2659 
2660  /* lock variables */
2661  if( SCIPisPositive(scip, val) )
2662  {
2663  for( v = nandvars - 1; v >= 0; --v )
2664  {
2665  SCIP_CALL( SCIPlockVarCons(scip, andvars[v], cons, FALSE, TRUE) );
2666  }
2667  }
2668  else
2669  {
2670  for( v = nandvars - 1; v >= 0; --v )
2671  {
2672  SCIP_CALL( SCIPlockVarCons(scip, andvars[v], cons, TRUE, FALSE) );
2673  }
2674  }
2675  }
2676  }
2677  else if( !SCIPisInfinity(scip, oldrhs) && SCIPisInfinity(scip, rhs) )
2678  {
2679  /* non-linear part */
2680  for( c = consdata->nconsanddatas - 1; c >= 0; --c )
2681  {
2682  CONSANDDATA* consanddata;
2683  SCIP_CONS* andcons;
2684 
2685  consanddata = consdata->consanddatas[c];
2686  assert(consanddata != NULL);
2687 
2688  andcons = consanddata->cons;
2689  assert(andcons != NULL);
2690 
2691  andvars = SCIPgetVarsAnd(scip, andcons);
2692  nandvars = SCIPgetNVarsAnd(scip, andcons);
2693  val = andnegs[c] ? -andcoefs[c] : andcoefs[c];
2694 
2695  /* lock variables */
2696  if( SCIPisPositive(scip, val) )
2697  {
2698  for( v = nandvars - 1; v >= 0; --v )
2699  {
2700  SCIP_CALL( SCIPunlockVarCons(scip, andvars[v], cons, FALSE, TRUE) );
2701  }
2702  }
2703  else
2704  {
2705  for( v = nandvars - 1; v >= 0; --v )
2706  {
2707  SCIP_CALL( SCIPunlockVarCons(scip, andvars[v], cons, TRUE, FALSE) );
2708  }
2709  }
2710  }
2711  }
2712  }
2713 
2714  /* check whether the right hand side is decreased, if and only if that's the case we maybe can propagate, tighten and add more cliques */
2715  if( SCIPisGT(scip, oldrhs, rhs) )
2716  {
2717  consdata->propagated = FALSE;
2718  }
2719 
2720  /* set new right hand side and update constraint data */
2721  SCIP_CALL( chgRhsLinearCons(scip, consdata->lincons, consdata->linconstype, rhs) );
2722  consdata->rhs = rhs;
2723  consdata->presolved = FALSE;
2724  consdata->changed = TRUE;
2725 
2726  /* free temporary memory */
2727  SCIPfreeBufferArray(scip, &andnegs);
2728  SCIPfreeBufferArray(scip, &andcoefs);
2729  SCIPfreeBufferArray(scip, &andress);
2730  SCIPfreeBufferArray(scip, &lincoefs);
2731  SCIPfreeBufferArray(scip, &linvars);
2732  SCIPfreeBufferArray(scip, &coefs);
2733  SCIPfreeBufferArray(scip, &vars);
2734 
2735  return SCIP_OKAY;
2736 }
2737 
2738 /** create and-constraints and get all and-resultants */
2739 static
2741  SCIP*const scip, /**< SCIP data structure */
2742  SCIP_CONSHDLR*const conshdlr, /**< pseudoboolean constraint handler */
2743  SCIP_VAR**const*const terms, /**< array of term variables to get and-constraints for */
2744  SCIP_Real*const termcoefs, /**< array of coefficients for and-constraints */
2745  int const nterms, /**< number of terms to get and-constraints for */
2746  int const*const ntermvars, /**< array of number of variable in each term */
2747  SCIP_Bool const initial, /**< should the LP relaxation of constraint be in the initial LP?
2748  * Usually set to TRUE. Set to FALSE for 'lazy constraints'. */
2749  SCIP_Bool const enforce, /**< should the constraint be enforced during node processing?
2750  * TRUE for model constraints, FALSE for additional, redundant
2751  * constraints. */
2752  SCIP_Bool const check, /**< should the constraint be checked for feasibility?
2753  * TRUE for model constraints, FALSE for additional, redundant
2754  * constraints. */
2755  SCIP_Bool const local, /**< is constraint only valid locally?
2756  * Usually set to FALSE. Has to be set to TRUE, e.g., for branching
2757  * constraints. */
2758  SCIP_Bool const modifiable, /**< is constraint modifiable (subject to column generation)?
2759  * Usually set to FALSE. In column generation applications, set to TRUE
2760  * if pricing adds coefficients to this constraint. */
2761  SCIP_Bool const dynamic, /**< is constraint subject to aging?
2762  * Usually set to FALSE. Set to TRUE for own cuts which
2763  * are seperated as constraints. */
2764  SCIP_Bool const stickingatnode, /**< should the constraint always be kept at the node where it was added, even
2765  * if it may be moved to a more global node?
2766  * Usually set to FALSE. Set to TRUE to for constraints that represent
2767  * node data. */
2768  SCIP_CONS**const andconss, /**< array to store all created and-constraints for given terms */
2769  SCIP_Real*const andvals, /**< array to store all coefficients of and-constraints */
2770  SCIP_Bool*const andnegs, /**< array to store negation status of and-constraints */
2771  int*const nandconss /**< number of created and constraints */
2772  )
2773 {
2774  int t;
2775 
2776  assert(scip != NULL);
2777  assert(conshdlr != NULL);
2778  assert(nterms == 0 || (terms != NULL && ntermvars != NULL));
2779  assert(andconss != NULL);
2780  assert(andvals != NULL);
2781  assert(nandconss != NULL);
2782 
2783  (*nandconss) = 0;
2784 
2785  if( nterms == 0 )
2786  return SCIP_OKAY;
2787 
2788  /* loop over all terms and create/get all and constraints */
2789  for( t = 0; t < nterms; ++t )
2790  {
2791  if( !SCIPisZero(scip, termcoefs[t]) && ntermvars[t] > 0 )
2792  {
2793  SCIP_CALL( createAndAddAndCons(scip, conshdlr, terms[t], ntermvars[t],
2794  initial, enforce, check, local, modifiable, dynamic, stickingatnode,
2795  &(andconss[*nandconss])) );
2796  assert(andconss[*nandconss] != NULL);
2797  andvals[*nandconss] = termcoefs[t];
2798  andnegs[*nandconss] = FALSE;
2799  ++(*nandconss);
2800  }
2801  }
2802 
2803  return SCIP_OKAY;
2804 }
2805 
2806 /** created linear constraint of pseudo boolean constraint */
2807 static
2809  SCIP*const scip, /**< SCIP data structure */
2810  SCIP_CONSHDLR*const conshdlr, /**< pseudoboolean constraint handler */
2811  SCIP_VAR**const linvars, /**< linear variables */
2812  int const nlinvars, /**< number of linear variables */
2813  SCIP_Real*const linvals, /**< linear coefficients */
2814  SCIP_VAR**const andress, /**< and-resultant variables */
2815  int const nandress, /**< number of and-resultant variables */
2816  SCIP_Real const*const andvals, /**< and-resultant coefficients */
2817  SCIP_Bool*const andnegs, /**< and-resultant negation status */
2818  SCIP_Real*const lhs, /**< pointer to left hand side of linear constraint */
2819  SCIP_Real*const rhs, /**< pointer to right hand side of linear constraint */
2820  SCIP_Bool const initial, /**< should the LP relaxation of constraint be in the initial LP?
2821  * Usually set to TRUE. Set to FALSE for 'lazy constraints'. */
2822  SCIP_Bool const separate, /**< should the constraint be separated during LP processing?
2823  * Usually set to TRUE. */
2824  SCIP_Bool const enforce, /**< should the constraint be enforced during node processing?
2825  * TRUE for model constraints, FALSE for additional, redundant
2826  * constraints. */
2827  SCIP_Bool const check, /**< should the constraint be checked for feasibility?
2828  * TRUE for model constraints, FALSE for additional, redundant
2829  * constraints. */
2830  SCIP_Bool const propagate, /**< should the constraint be propagated during node processing?
2831  * Usually set to TRUE. */
2832  SCIP_Bool const local, /**< is constraint only valid locally?
2833  * Usually set to FALSE. Has to be set to TRUE, e.g., for branching
2834  * constraints. */
2835  SCIP_Bool const modifiable, /**< is constraint modifiable (subject to column generation)?
2836  * Usually set to FALSE. In column generation applications, set to TRUE
2837  * if pricing adds coefficients to this constraint. */
2838  SCIP_Bool const dynamic, /**< is constraint subject to aging?
2839  * Usually set to FALSE. Set to TRUE for own cuts which
2840  * are seperated as constraints. */
2841  SCIP_Bool const removable, /**< should the relaxation be removed from the LP due to aging or cleanup?
2842  * Usually set to FALSE. Set to TRUE for 'lazy constraints' and 'user
2843  * cuts'. */
2844  SCIP_Bool const stickingatnode, /**< should the constraint always be kept at the node where it was added, even
2845  * if it may be moved to a more global node?
2846  * Usually set to FALSE. Set to TRUE to for constraints that represent
2847  * node data. */
2848  SCIP_CONS**const lincons, /**< pointer to store created linear constraint */
2849  SCIP_LINEARCONSTYPE*const linconstype /**< pointer to store the type of the linear constraint */
2850  )
2851 {
2852  SCIP_CONSHDLRDATA* conshdlrdata;
2853  SCIP_CONSHDLR* upgrconshdlr;
2854  SCIP_CONS* cons;
2855  char name[SCIP_MAXSTRLEN];
2856  int v;
2857  SCIP_Bool created;
2858  SCIP_Bool integral;
2859  int nzero;
2860  int ncoeffspone;
2861  int ncoeffsnone;
2862  int ncoeffspint;
2863  int ncoeffsnint;
2864 
2865  assert(scip != NULL);
2866  assert(conshdlr != NULL);
2867  assert(nlinvars == 0 || (linvars != NULL && linvals != NULL));
2868  assert(nandress == 0 || (andress != NULL && andvals != NULL));
2869  assert(lhs != NULL);
2870  assert(rhs != NULL);
2871  assert(lincons != NULL);
2872  assert(linconstype != NULL);
2873  assert(nlinvars > 0 || nandress > 0);
2874 
2875  conshdlrdata = SCIPconshdlrGetData(conshdlr);
2876  assert(conshdlrdata != NULL);
2877 
2878  (*linconstype) = SCIP_LINEARCONSTYPE_INVALIDCONS;
2879  (*lincons) = NULL;
2880  cons = NULL;
2881 
2882  (void)SCIPsnprintf(name, SCIP_MAXSTRLEN, "pseudoboolean_linear%d", conshdlrdata->nlinconss);
2883  ++(conshdlrdata->nlinconss);
2884 
2885  created = FALSE;
2886 
2887  if( !modifiable )
2888  {
2889  SCIP_Real val;
2890  int nvars;
2891 
2892  /* calculate some statistics for upgrading on linear constraint */
2893  nzero = 0;
2894  ncoeffspone = 0;
2895  ncoeffsnone = 0;
2896  ncoeffspint = 0;
2897  ncoeffsnint = 0;
2898  integral = TRUE;
2899  nvars = nlinvars + nandress;
2900 
2901  /* calculate information over linear part */
2902  for( v = nlinvars - 1; v >= 0; --v )
2903  {
2904  val = linvals[v];
2905 
2906  if( SCIPisZero(scip, val) )
2907  {
2908  ++nzero;
2909  continue;
2910  }
2911  if( SCIPisEQ(scip, val, 1.0) )
2912  ++ncoeffspone;
2913  else if( SCIPisEQ(scip, val, -1.0) )
2914  ++ncoeffsnone;
2915  else if( SCIPisIntegral(scip, val) )
2916  {
2917  if( SCIPisPositive(scip, val) )
2918  ++ncoeffspint;
2919  else
2920  ++ncoeffsnint;
2921  }
2922  else
2923  {
2924  integral = FALSE;
2925  break;
2926  }
2927  }
2928 
2929  if( integral )
2930  {
2931  /* calculate information over and-resultants */
2932  for( v = nandress - 1; v >= 0; --v )
2933  {
2934  val = andvals[v];
2935 
2936  if( SCIPisZero(scip, val) )
2937  {
2938  ++nzero;
2939  continue;
2940  }
2941  if( SCIPisEQ(scip, val, 1.0) )
2942  ++ncoeffspone;
2943  else if( SCIPisEQ(scip, val, -1.0) )
2944  ++ncoeffsnone;
2945  else if( SCIPisIntegral(scip, val) )
2946  {
2947  if( SCIPisPositive(scip, val) )
2948  ++ncoeffspint;
2949  else
2950  ++ncoeffsnint;
2951  }
2952  else
2953  {
2954  integral = FALSE;
2955  break;
2956  }
2957  }
2958  }
2959 
2960  SCIPdebugMsg(scip, "While creating the linear constraint of the pseudoboolean constraint we found %d zero coefficients that were removed\n", nzero);
2961 
2962  /* try to upgrade to a special linear constraint */
2963  if( integral )
2964  {
2965  upgrconshdlr = SCIPfindConshdlr(scip, "logicor");
2966 
2967  /* check, if linear constraint can be upgraded to logic or constraint
2968  * - logic or constraints consist only of binary variables with a
2969  * coefficient of +1.0 or -1.0 (variables with -1.0 coefficients can be negated):
2970  * lhs <= x1 + ... + xp - y1 - ... - yn <= rhs
2971  * - negating all variables y = (1-Y) with negative coefficients gives:
2972  * lhs + n <= x1 + ... + xp + Y1 + ... + Yn <= rhs + n
2973  * - negating all variables x = (1-X) with positive coefficients and multiplying with -1 gives:
2974  * p - rhs <= X1 + ... + Xp + y1 + ... + yn <= p - lhs
2975  * - logic or constraints have left hand side of +1.0, and right hand side of +infinity: x(S) >= 1.0
2976  * -> without negations: (lhs == 1 - n and rhs == +inf) or (lhs == -inf and rhs = p - 1)
2977  */
2978  if( upgrconshdlr != NULL && nvars > 2 && ncoeffspone + ncoeffsnone == nvars
2979  && ((SCIPisEQ(scip, *lhs, 1.0 - ncoeffsnone) && SCIPisInfinity(scip, *rhs))
2980  || (SCIPisInfinity(scip, -*lhs) && SCIPisEQ(scip, *rhs, ncoeffspone - 1.0))) )
2981  {
2982  SCIP_VAR** transvars;
2983  int mult;
2984 
2985  SCIPdebugMsg(scip, "linear constraint will be logic-or constraint\n");
2986 
2987  /* check, if we have to multiply with -1 (negate the positive vars) or with +1 (negate the negative vars) */
2988  mult = SCIPisInfinity(scip, *rhs) ? +1 : -1;
2989 
2990  /* get temporary memory */
2991  SCIP_CALL( SCIPallocBufferArray(scip, &transvars, nvars) );
2992 
2993  /* negate positive or negative variables */
2994  for( v = 0; v < nlinvars; ++v )
2995  {
2996  if( mult * linvals[v] > 0.0 )
2997  transvars[v] = linvars[v];
2998  else
2999  {
3000  SCIP_CALL( SCIPgetNegatedVar(scip, linvars[v], &transvars[v]) );
3001  }
3002  assert(transvars[v] != NULL);
3003  }
3004 
3005  /* negate positive or negative variables */
3006  for( v = 0; v < nandress; ++v )
3007  {
3008  if( mult * andvals[v] > 0.0 )
3009  transvars[nlinvars + v] = andress[v];
3010  else
3011  {
3012  SCIP_CALL( SCIPgetNegatedVar(scip, andress[v], &transvars[nlinvars + v]) );
3013  andnegs[v] = TRUE;
3014  }
3015  assert(transvars[nlinvars + v] != NULL);
3016  }
3017 
3018  assert(!modifiable);
3019  /* create the constraint */
3020  SCIP_CALL( SCIPcreateConsLogicor(scip, &cons, name, nvars, transvars,
3021  initial, separate, enforce, check, propagate, local, modifiable, dynamic, removable, stickingatnode) );
3022 
3023  created = TRUE;
3024  (*linconstype) = SCIP_LINEARCONSTYPE_LOGICOR;
3025 
3026  /* free temporary memory */
3027  SCIPfreeBufferArray(scip, &transvars);
3028 
3029  *lhs = 1.0;
3030  *rhs = SCIPinfinity(scip);
3031  }
3032 
3033  upgrconshdlr = SCIPfindConshdlr(scip, "setppc");
3034 
3035  /* check, if linear constraint can be upgraded to set partitioning, packing, or covering constraint
3036  * - all set partitioning / packing / covering constraints consist only of binary variables with a
3037  * coefficient of +1.0 or -1.0 (variables with -1.0 coefficients can be negated):
3038  * lhs <= x1 + ... + xp - y1 - ... - yn <= rhs
3039  * - negating all variables y = (1-Y) with negative coefficients gives:
3040  * lhs + n <= x1 + ... + xp + Y1 + ... + Yn <= rhs + n
3041  * - negating all variables x = (1-X) with positive coefficients and multiplying with -1 gives:
3042  * p - rhs <= X1 + ... + Xp + y1 + ... + yn <= p - lhs
3043  * - a set partitioning constraint has left hand side of +1.0, and right hand side of +1.0 : x(S) == 1.0
3044  * -> without negations: lhs == rhs == 1 - n or lhs == rhs == p - 1
3045  * - a set packing constraint has left hand side of -infinity, and right hand side of +1.0 : x(S) <= 1.0
3046  * -> without negations: (lhs == -inf and rhs == 1 - n) or (lhs == p - 1 and rhs = +inf)
3047  * - a set covering constraint has left hand side of +1.0, and right hand side of +infinity: x(S) >= 1.0
3048  * -> without negations: (lhs == 1 - n and rhs == +inf) or (lhs == -inf and rhs = p - 1)
3049  */
3050  if( upgrconshdlr != NULL && !created && ncoeffspone + ncoeffsnone == nvars )
3051  {
3052  SCIP_VAR** transvars;
3053  int mult;
3054 
3055  if( SCIPisEQ(scip, *lhs, *rhs) && (SCIPisEQ(scip, *lhs, 1.0 - ncoeffsnone) || SCIPisEQ(scip, *lhs, ncoeffspone - 1.0)) )
3056  {
3057  SCIPdebugMsg(scip, "linear pseudoboolean constraint will be a set partitioning constraint\n");
3058 
3059  /* check, if we have to multiply with -1 (negate the positive vars) or with +1 (negate the negative vars) */
3060  mult = SCIPisEQ(scip, *lhs, 1.0 - ncoeffsnone) ? +1 : -1;
3061 
3062  /* get temporary memory */
3063  SCIP_CALL( SCIPallocBufferArray(scip, &transvars, nvars) );
3064 
3065  /* negate positive or negative variables for linear variables */
3066  for( v = 0; v < nlinvars; ++v )
3067  {
3068  if( mult * linvals[v] > 0.0 )
3069  transvars[v] = linvars[v];
3070  else
3071  {
3072  SCIP_CALL( SCIPgetNegatedVar(scip, linvars[v], &transvars[v]) );
3073  }
3074  assert(transvars[v] != NULL);
3075  }
3076 
3077  /* negate positive or negative variables for and-resultants */
3078  for( v = 0; v < nandress; ++v )
3079  {
3080  if( mult * andvals[v] > 0.0 )
3081  transvars[nlinvars + v] = andress[v];
3082  else
3083  {
3084  SCIP_CALL( SCIPgetNegatedVar(scip, andress[v], &transvars[nlinvars + v]) );
3085  andnegs[v] = TRUE;
3086  }
3087  assert(transvars[nlinvars + v] != NULL);
3088  }
3089 
3090  /* create the constraint */
3091  assert(!modifiable);
3092  SCIP_CALL( SCIPcreateConsSetpart(scip, &cons, name, nvars, transvars,
3093  initial, separate, enforce, check, propagate, local, modifiable, dynamic, removable, stickingatnode) );
3094 
3095  created = TRUE;
3096  (*linconstype) = SCIP_LINEARCONSTYPE_SETPPC;
3097 
3098  /* release temporary memory */
3099  SCIPfreeBufferArray(scip, &transvars);
3100 
3101  *lhs = 1.0;
3102  *rhs = 1.0;
3103  }
3104  else if( (SCIPisInfinity(scip, -*lhs) && SCIPisEQ(scip, *rhs, 1.0 - ncoeffsnone))
3105  || (SCIPisEQ(scip, *lhs, ncoeffspone - 1.0) && SCIPisInfinity(scip, *rhs)) )
3106  {
3107  SCIPdebugMsg(scip, "linear pseudoboolean constraint will be a set packing constraint\n");
3108 
3109  /* check, if we have to multiply with -1 (negate the positive vars) or with +1 (negate the negative vars) */
3110  mult = SCIPisInfinity(scip, -*lhs) ? +1 : -1;
3111 
3112  /* get temporary memory */
3113  SCIP_CALL( SCIPallocBufferArray(scip, &transvars, nvars) );
3114 
3115  /* negate positive or negative variables for linear variables */
3116  for( v = 0; v < nlinvars; ++v )
3117  {
3118  if( mult * linvals[v] > 0.0 )
3119  transvars[v] = linvars[v];
3120  else
3121  {
3122  SCIP_CALL( SCIPgetNegatedVar(scip, linvars[v], &transvars[v]) );
3123  }
3124  assert(transvars[v] != NULL);
3125  }
3126 
3127  /* negate positive or negative variables for and-resultants*/
3128  for( v = 0; v < nandress; ++v )
3129  {
3130  if( mult * andvals[v] > 0.0 )
3131  transvars[nlinvars + v] = andress[v];
3132  else
3133  {
3134  SCIP_CALL( SCIPgetNegatedVar(scip, andress[v], &transvars[nlinvars + v]) );
3135  andnegs[v] = TRUE;
3136  }
3137  assert(transvars[nlinvars + v] != NULL);
3138  }
3139 
3140  /* create the constraint */
3141  assert(!modifiable);
3142  SCIP_CALL( SCIPcreateConsSetpack(scip, &cons, name, nvars, transvars,
3143  initial, separate, enforce, check, propagate, local, modifiable, dynamic, removable, stickingatnode) );
3144 
3145  created = TRUE;
3146  (*linconstype) = SCIP_LINEARCONSTYPE_SETPPC;
3147 
3148  /* release temporary memory */
3149  SCIPfreeBufferArray(scip, &transvars);
3150 
3151  *lhs = -SCIPinfinity(scip);
3152  *rhs = 1.0;
3153  }
3154  else if( (SCIPisEQ(scip, *lhs, 1.0 - ncoeffsnone) && SCIPisInfinity(scip, *rhs))
3155  || (SCIPisInfinity(scip, -*lhs) && SCIPisEQ(scip, *rhs, ncoeffspone - 1.0)) )
3156  {
3157  if( nvars != 1 )
3158  {
3159  if( nvars == 2 )
3160  {
3161  SCIPwarningMessage(scip, "Does not expect this, because this constraint should be a set packing constraint.\n");
3162  }
3163  else
3164  {
3165  SCIPwarningMessage(scip, "Does not expect this, because this constraint should be a logicor constraint.\n");
3166  }
3167  }
3168  SCIPdebugMsg(scip, "linear pseudoboolean constraint will be a set covering constraint\n");
3169 
3170  /* check, if we have to multiply with -1 (negate the positive vars) or with +1 (negate the negative vars) */
3171  mult = SCIPisInfinity(scip, *rhs) ? +1 : -1;
3172 
3173  /* get temporary memory */
3174  SCIP_CALL( SCIPallocBufferArray(scip, &transvars, nvars) );
3175 
3176  /* negate positive or negative variables for linear variables */
3177  for( v = 0; v < nlinvars; ++v )
3178  {
3179  if( mult * linvals[v] > 0.0 )
3180  transvars[v] = linvars[v];
3181  else
3182  {
3183  SCIP_CALL( SCIPgetNegatedVar(scip, linvars[v], &transvars[v]) );
3184  }
3185  assert(transvars[v] != NULL);
3186  }
3187 
3188  /* negate positive or negative variables for and-resultants*/
3189  for( v = 0; v < nandress; ++v )
3190  {
3191  if( mult * andvals[v] > 0.0 )
3192  transvars[nlinvars + v] = andress[v];
3193  else
3194  {
3195  SCIP_CALL( SCIPgetNegatedVar(scip, andress[v], &transvars[nlinvars + v]) );
3196  andnegs[v] = TRUE;
3197  }
3198  assert(transvars[nlinvars + v] != NULL);
3199  }
3200 
3201  /* create the constraint */
3202  assert(!modifiable);
3203  SCIP_CALL( SCIPcreateConsSetcover(scip, &cons, name, nvars, transvars,
3204  initial, separate, enforce, check, propagate, local, modifiable, dynamic, removable, stickingatnode) );
3205 
3206  created = TRUE;
3207  (*linconstype) = SCIP_LINEARCONSTYPE_SETPPC;
3208 
3209  /* release temporary memory */
3210  SCIPfreeBufferArray(scip, &transvars);
3211 
3212  *lhs = 1.0;
3213  *rhs = SCIPinfinity(scip);
3214  }
3215  }
3216 
3217  upgrconshdlr = SCIPfindConshdlr(scip, "knapsack");
3218 
3219  /* check, if linear constraint can be upgraded to a knapsack constraint
3220  * - all variables must be binary
3221  * - all coefficients must be integral
3222  * - exactly one of the sides must be infinite
3223  */
3224  if( upgrconshdlr != NULL && !created && (ncoeffspone + ncoeffsnone + ncoeffspint + ncoeffsnint == nvars) && (SCIPisInfinity(scip, -*lhs) != SCIPisInfinity(scip, *rhs)) )
3225  {
3226  SCIP_VAR** transvars;
3227  SCIP_Longint* weights;
3228  SCIP_Longint capacity;
3229  SCIP_Longint weight;
3230  int mult;
3231 
3232  SCIPdebugMsg(scip, "linear pseudoboolean constraint will be a knapsack constraint\n");
3233 
3234  /* get temporary memory */
3235  SCIP_CALL( SCIPallocBufferArray(scip, &transvars, nvars) );
3236  SCIP_CALL( SCIPallocBufferArray(scip, &weights, nvars) );
3237 
3238  /* if the right hand side is non-infinite, we have to negate all variables with negative coefficient;
3239  * otherwise, we have to negate all variables with positive coefficient and multiply the row with -1
3240  */
3241  if( SCIPisInfinity(scip, *rhs) )
3242  {
3243  mult = -1;
3244  capacity = (SCIP_Longint)SCIPfeasFloor(scip, -*lhs);
3245  }
3246  else
3247  {
3248  mult = +1;
3249  capacity = (SCIP_Longint)SCIPfeasFloor(scip, *rhs);
3250  }
3251 
3252  /* negate positive or negative variables for linear variables */
3253  for( v = 0; v < nlinvars; ++v )
3254  {
3255  assert(SCIPisFeasIntegral(scip, linvals[v]));
3256  weight = mult * (SCIP_Longint)SCIPfeasFloor(scip, linvals[v]);
3257  if( weight > 0 )
3258  {
3259  transvars[v] = linvars[v];
3260  weights[v] = weight;
3261  }
3262  else
3263  {
3264  SCIP_CALL( SCIPgetNegatedVar(scip, linvars[v], &transvars[v]) );
3265  weights[v] = -weight;
3266  capacity -= weight;
3267  }
3268  assert(transvars[v] != NULL);
3269  }
3270  /* negate positive or negative variables for and-resultants */
3271  for( v = 0; v < nandress; ++v )
3272  {
3273  assert(SCIPisFeasIntegral(scip, andvals[v]));
3274  weight = mult * (SCIP_Longint)SCIPfeasFloor(scip, andvals[v]);
3275  if( weight > 0 )
3276  {
3277  transvars[nlinvars + v] = andress[v];
3278  weights[nlinvars + v] = weight;
3279  }
3280  else
3281  {
3282  SCIP_CALL( SCIPgetNegatedVar(scip, andress[v], &transvars[nlinvars + v]) );
3283  andnegs[v] = TRUE;
3284  weights[nlinvars + v] = -weight;
3285  capacity -= weight;
3286  }
3287  assert(transvars[nlinvars + v] != NULL);
3288  }
3289 
3290  /* create the constraint */
3291  SCIP_CALL( SCIPcreateConsKnapsack(scip, &cons, name, nvars, transvars, weights, capacity,
3292  initial, separate, enforce, check, propagate, local, modifiable, dynamic, removable, stickingatnode) );
3293 
3294  created = TRUE;
3295  (*linconstype) = SCIP_LINEARCONSTYPE_KNAPSACK;
3296 
3297  /* free temporary memory */
3298  SCIPfreeBufferArray(scip, &weights);
3299  SCIPfreeBufferArray(scip, &transvars);
3300 
3301  *lhs = -SCIPinfinity(scip);
3302  *rhs = capacity;
3303  }
3304 #ifdef WITHEQKNAPSACK
3305 
3306  upgrconshdlr = SCIPfindConshdlr(scip, "eqknapsack");
3307 
3308  /* check, if linear constraint can be upgraded to a knapsack constraint
3309  * - all variables must be binary
3310  * - all coefficients must be integral
3311  * - both sides must be infinite
3312  */
3313  if( upgrconshdlr != NULL && !created && (ncoeffspone + ncoeffsnone + ncoeffspint + ncoeffsnint == nvars) && SCIPisEQ(scip, *lhs, *rhs) )
3314  {
3315  SCIP_VAR** transvars;
3316  SCIP_Longint* weights;
3317  SCIP_Longint capacity;
3318  SCIP_Longint weight;
3319  int mult;
3320 
3321  assert(!SCIPisInfinity(scip, *rhs));
3322 
3323  SCIPdebugMsg(scip, "linear pseudoboolean constraint will be a equality-knapsack constraint\n");
3324 
3325  /* get temporary memory */
3326  SCIP_CALL( SCIPallocBufferArray(scip, &transvars, nvars) );
3327  SCIP_CALL( SCIPallocBufferArray(scip, &weights, nvars) );
3328 
3329  if( SCIPisPositive(scip, *rhs) )
3330  {
3331  mult = +1;
3332  capacity = (SCIP_Longint)SCIPfeasFloor(scip, *rhs);
3333  }
3334  else
3335  {
3336  mult = -1;
3337  capacity = (SCIP_Longint)SCIPfeasFloor(scip, -*rhs);
3338  }
3339 
3340  /* negate positive or negative variables for linear variables */
3341  for( v = 0; v < nlinvars; ++v )
3342  {
3343  assert(SCIPisFeasIntegral(scip, linvals[v]));
3344  weight = mult * (SCIP_Longint)SCIPfeasFloor(scip, linvals[v]);
3345  if( weight > 0 )
3346  {
3347  transvars[v] = linvars[v];
3348  weights[v] = weight;
3349  }
3350  else
3351  {
3352  SCIP_CALL( SCIPgetNegatedVar(scip, linvars[v], &transvars[v]) );
3353  weights[v] = -weight;
3354  capacity -= weight;
3355  }
3356  assert(transvars[v] != NULL);
3357  }
3358  /* negate positive or negative variables for and-resultants */
3359  for( v = 0; v < nandress; ++v )
3360  {
3361  assert(SCIPisFeasIntegral(scip, andvals[v]));
3362  weight = mult * (SCIP_Longint)SCIPfeasFloor(scip, andvals[v]);
3363  if( weight > 0 )
3364  {
3365  transvars[nlinvars + v] = andress[v];
3366  weights[nlinvars + v] = weight;
3367  }
3368  else
3369  {
3370  SCIP_CALL( SCIPgetNegatedVar(scip, andress[v], &transvars[nlinvars + v]) );
3371  andnegs[v] = TRUE;
3372  weights[nlinvars + v] = -weight;
3373  capacity -= weight;
3374  }
3375  assert(transvars[nlinvars + v] != NULL);
3376  }
3377 
3378  /* create the constraint */
3379  SCIP_CALL( SCIPcreateConsEqKnapsack(scip, &cons, name, nvars, transvars, weights, capacity,
3380  initial, separate, enforce, check, propagate, local, modifiable, dynamic, removable, stickingatnode) );
3381 
3382  created = TRUE;
3383  (*linconstype) = SCIP_LINEARCONSTYPE_EQKNAPSACK;
3384 
3385  /* free temporary memory */
3386  SCIPfreeBufferArray(scip, &weights);
3387  SCIPfreeBufferArray(scip, &transvars);
3388 
3389  *lhs = capacity;
3390  *rhs = capacity;
3391  }
3392 #endif
3393  }
3394  }
3395 
3396  upgrconshdlr = SCIPfindConshdlr(scip, "linear");
3397  assert(created || upgrconshdlr != NULL);
3398 
3399  if( !created )
3400  {
3401  SCIP_CALL( SCIPcreateConsLinear(scip, &cons, name, nlinvars, linvars, linvals, *lhs, *rhs,
3402  initial, separate, enforce, check, propagate, local, modifiable, dynamic, removable, stickingatnode) );
3403 
3404  (*linconstype) = SCIP_LINEARCONSTYPE_LINEAR;
3405 
3406  /* add all and-resultants */
3407  for( v = 0; v < nandress; ++v )
3408  {
3409  assert(andress[v] != NULL);
3410 
3411  /* add auxiliary variables to linear constraint */
3412  SCIP_CALL( SCIPaddCoefLinear(scip, cons, andress[v], andvals[v]) );
3413  }
3414  }
3415 
3416  assert(cons != NULL && *linconstype > SCIP_LINEARCONSTYPE_INVALIDCONS);
3417 
3418  SCIP_CALL( SCIPaddCons(scip, cons) );
3419  SCIPdebugPrintCons(scip, cons, NULL);
3420 
3421  *lincons = cons;
3422  SCIP_CALL( SCIPcaptureCons(scip, *lincons) );
3423 
3424  /* mark linear constraint not to be upgraded - otherwise we loose control over it */
3425  SCIPconsAddUpgradeLocks(cons, 1);
3426 
3427  SCIP_CALL( SCIPreleaseCons(scip, &cons) );
3428 
3429  return SCIP_OKAY;
3430 }
3431 
3432 /** checks one original pseudoboolean constraint for feasibility of given solution */
3433 static
3435  SCIP*const scip, /**< SCIP data structure */
3436  SCIP_CONS*const cons, /**< pseudo boolean constraint */
3437  SCIP_SOL*const sol, /**< solution to be checked, or NULL for current solution */
3438  SCIP_Bool*const violated, /**< pointer to store whether the constraint is violated */
3439  SCIP_Bool const printreason /**< should violation of constraint be printed */
3440  )
3441 {
3442  SCIP_CONSDATA* consdata;
3443  SCIP_CONSHDLR* conshdlr;
3444  SCIP_CONSHDLRDATA* conshdlrdata;
3445 
3446  SCIP_VAR** vars;
3447  SCIP_Real* coefs;
3448  int nvars;
3449  SCIP_Real lhs;
3450  SCIP_Real rhs;
3451 
3452  SCIP_VAR** linvars;
3453  SCIP_Real* lincoefs;
3454  int nlinvars;
3455  int v;
3456 
3457  SCIP_VAR** andress;
3458  SCIP_Real* andcoefs;
3459  int nandress;
3460 
3461  SCIP_CONS* andcons;
3462  SCIP_Real andvalue;
3463  SCIP_Real activity;
3464  int c;
3465 
3466  SCIP_Real lhsviol;
3467  SCIP_Real rhsviol;
3468  SCIP_Real absviol;
3469  SCIP_Real relviol;
3470 
3471  assert(scip != NULL);
3472  assert(cons != NULL);
3473  assert(SCIPconsIsOriginal(cons));
3474  assert(violated != NULL);
3475 
3476  *violated = FALSE;
3477 
3478  SCIPdebugMsg(scip, "checking original pseudo boolean constraint <%s>\n", SCIPconsGetName(cons));
3479  SCIPdebugPrintCons(scip, cons, NULL);
3480 
3481  consdata = SCIPconsGetData(cons);
3482  assert(consdata != NULL);
3483  assert(consdata->lincons != NULL);
3484  assert(consdata->linconstype > SCIP_LINEARCONSTYPE_INVALIDCONS);
3485  assert(SCIPconsIsOriginal(consdata->lincons));
3486 
3487  /* gets number of variables in linear constraint */
3488  SCIP_CALL( getLinearConsNVars(scip, consdata->lincons, consdata->linconstype, &nvars) );
3489 
3490  /* allocate temporary memory */
3491  SCIP_CALL( SCIPallocBufferArray(scip, &vars, nvars) );
3492  SCIP_CALL( SCIPallocBufferArray(scip, &coefs, nvars) );
3493  SCIP_CALL( SCIPallocBufferArray(scip, &linvars, nvars) );
3494  SCIP_CALL( SCIPallocBufferArray(scip, &lincoefs, nvars) );
3495  SCIP_CALL( SCIPallocBufferArray(scip, &andress, nvars) );
3496  SCIP_CALL( SCIPallocBufferArray(scip, &andcoefs, nvars) );
3497 
3498  /* get sides of linear constraint */
3499  SCIP_CALL( getLinearConsSides(scip, consdata->lincons, consdata->linconstype, &lhs, &rhs) );
3500  assert(!SCIPisInfinity(scip, lhs));
3501  assert(!SCIPisInfinity(scip, -rhs));
3502  assert(SCIPisLE(scip, lhs, rhs));
3503 
3504  /* get variables and coefficient of linear constraint */
3505  SCIP_CALL( getLinearConsVarsData(scip, consdata->lincons, consdata->linconstype, vars, coefs, &nvars) );
3506  assert(nvars == 0 || (coefs != NULL));
3507 
3508  /* number of variables should be consistent, number of 'real' linear variables plus number of and-constraints should
3509  * have to be equal to the number of variables in the linear constraint
3510  */
3511  assert(consdata->nlinvars + consdata->nconsanddatas == nvars);
3512 
3513  nlinvars = 0;
3514 
3515  conshdlr = SCIPconsGetHdlr(cons);
3516  assert(conshdlr != NULL);
3517  conshdlrdata = SCIPconshdlrGetData(conshdlr);
3518  assert(conshdlrdata != NULL);
3519  assert(conshdlrdata->hashmap != NULL);
3520 
3521  nandress = 0;
3522 
3523  activity = 0.0;
3524 
3525  /* split variables into original and artificial variables and compute activity on normal linear variables (without
3526  * terms)
3527  */
3528  for( v = 0; v < nvars; ++v )
3529  {
3530  SCIP_VAR* hashmapvar;
3531  SCIP_Bool negated;
3532 
3533  assert(vars[v] != NULL);
3534 
3535  /* negated variables can also exist in the original problem, so we need to check */
3536  if( !SCIPhashmapExists(conshdlrdata->hashmap, (void*)(vars[v])) && SCIPvarIsNegated(vars[v]) )
3537  {
3538  hashmapvar = SCIPvarGetNegationVar(vars[v]);
3539  negated = TRUE;
3540  }
3541  else
3542  {
3543  hashmapvar = vars[v];
3544  negated = FALSE;
3545  }
3546  assert(hashmapvar != NULL);
3547 
3548  if( !SCIPhashmapExists(conshdlrdata->hashmap, (void*)(hashmapvar)) )
3549  {
3550  assert(!SCIPhashmapExists(conshdlrdata->hashmap, (void*)(vars[v])));
3551 
3552  activity += coefs[v] * SCIPgetSolVal(scip, sol, vars[v]);
3553 
3554  linvars[nlinvars] = vars[v];
3555  lincoefs[nlinvars] = coefs[v];
3556  ++nlinvars;
3557  }
3558  else
3559  {
3560  /* negate coefficient in case of an original negated variable */
3561  andress[nandress] = hashmapvar;
3562  if( negated )
3563  {
3564  if( !SCIPisInfinity(scip, -lhs) )
3565  lhs -= coefs[v];
3566  if( !SCIPisInfinity(scip, rhs) )
3567  rhs -= coefs[v];
3568  andcoefs[nandress] = -coefs[v];
3569  }
3570  else
3571  andcoefs[nandress] = coefs[v];
3572  ++nandress;
3573  }
3574  }
3575  assert(nandress == consdata->nconsanddatas);
3576 
3577  SCIPsortPtrReal((void**)andress, andcoefs, SCIPvarComp, nandress);
3578 
3579  SCIPdebugMsg(scip, "nlinvars = %d, nandress = %d\n", nlinvars, nandress);
3580  SCIPdebugMsg(scip, "linear activity = %g\n", activity);
3581 
3582  /* compute and add solution values on terms */
3583  for( c = consdata->nconsanddatas - 1; c >= 0; --c )
3584  {
3585  SCIP_VAR** andvars;
3586  int nandvars;
3587 #ifndef NDEBUG
3588  SCIP_VAR* res;
3589 #endif
3590  andcons = consdata->consanddatas[c]->origcons;
3591 
3592  /* if after during or before presolving a solution will be transformed into original space and will be checked
3593  * there, but origcons was already removed and only the pointer to the transformed and-constraint is existing
3594  */
3595  if( andcons == NULL )
3596  {
3597  andcons = consdata->consanddatas[c]->cons;
3598  }
3599  assert(andcons != NULL);
3600 
3601  andvars = SCIPgetVarsAnd(scip, andcons);
3602  nandvars = SCIPgetNVarsAnd(scip, andcons);
3603 
3604 #ifndef NDEBUG
3605  res = SCIPgetResultantAnd(scip, andcons);
3606  assert(nandvars == 0 || (andvars != NULL && res != NULL));
3607  assert(res == andress[c]);
3608 #endif
3609 
3610  andvalue = 1;
3611  /* check if the and-constraint is violated */
3612  for( v = nandvars - 1; v >= 0; --v )
3613  {
3614  andvalue *= SCIPgetSolVal(scip, sol, andvars[v]);
3615  if( SCIPisFeasZero(scip, andvalue) )
3616  break;
3617  }
3618  activity += andvalue * andcoefs[c];
3619  }
3620  SCIPdebugMsg(scip, "lhs = %g, overall activity = %g, rhs = %g\n", lhs, activity, rhs);
3621 
3622  /* calculate absolute and relative violation */
3623  lhsviol = lhs - activity;
3624  rhsviol = activity - rhs;
3625 
3626  if(lhsviol > rhsviol)
3627  {
3628  absviol = lhsviol;
3629  relviol = SCIPrelDiff(lhs, activity);
3630  }
3631  else
3632  {
3633  absviol = rhsviol;
3634  relviol = SCIPrelDiff(activity, rhs);
3635  }
3636 
3637  /* update absolute and relative violation of the solution */
3638  if( sol != NULL )
3639  SCIPupdateSolConsViolation(scip, sol, absviol, relviol);
3640 
3641  /* check left hand side for violation */
3642  if( SCIPisFeasLT(scip, activity, lhs) )
3643  {
3644  if( printreason )
3645  {
3646  SCIP_CALL( SCIPprintCons(scip, cons, NULL ) );
3647  SCIPinfoMessage(scip, NULL, ";\n");
3648  SCIPinfoMessage(scip, NULL, "violation: left hand side is violated by %.15g\n", lhs - activity);
3649 
3650  /* print linear constraint in SCIP_DEBUG mode too */
3651  SCIPdebugPrintCons(scip, SCIPconsGetData(cons)->lincons, NULL);
3652  }
3653 
3654  *violated = TRUE;
3655  }
3656 
3657  /* check right hand side for violation */
3658  if( SCIPisFeasGT(scip, activity, rhs) )
3659  {
3660  if( printreason )
3661  {
3662  SCIP_CALL( SCIPprintCons(scip, cons, NULL ) );
3663  SCIPinfoMessage(scip, NULL, ";\n");
3664  SCIPinfoMessage(scip, NULL, "violation: right hand side is violated by %.15g\n", activity - rhs);
3665  }
3666 
3667  *violated = TRUE;
3668  }
3669 
3670  /* free temporary memory */
3671  SCIPfreeBufferArray(scip, &andcoefs);
3672  SCIPfreeBufferArray(scip, &andress);
3673  SCIPfreeBufferArray(scip, &lincoefs);
3674  SCIPfreeBufferArray(scip, &linvars);
3675  SCIPfreeBufferArray(scip, &coefs);
3676  SCIPfreeBufferArray(scip, &vars);
3677 
3678  return SCIP_OKAY;
3679 }
3680 
3681 /** checks all and-constraints inside the pseudoboolean constraint handler for feasibility of given solution or current
3682  * solution
3683  */
3684 static
3686  SCIP*const scip, /**< SCIP data structure */
3687  SCIP_CONSHDLR*const conshdlr, /**< pseudo boolean constraint handler */
3688  SCIP_SOL*const sol, /**< solution to be checked, or NULL for current solution */
3689  SCIP_Bool*const violated /**< pointer to store whether the constraint is violated */
3690  )
3691 {
3692  SCIP_CONSHDLRDATA* conshdlrdata;
3693  SCIP_CONS* andcons;
3694  SCIP_VAR** vars;
3695  SCIP_VAR* res;
3696  int nvars;
3697  SCIP_Real andvalue;
3698  int c;
3699  int v;
3700 
3701  assert(scip != NULL);
3702  assert(conshdlr != NULL);
3703  assert(violated != NULL);
3704 
3705  conshdlrdata = SCIPconshdlrGetData(conshdlr);
3706  assert(conshdlrdata != NULL);
3707 
3708  *violated = FALSE;
3709 
3710  for( c = conshdlrdata->nallconsanddatas - 1; c >= 0; --c )
3711  {
3712  if( !conshdlrdata->allconsanddatas[c]->istransformed )
3713  continue;
3714 
3715  andcons = conshdlrdata->allconsanddatas[c]->cons;
3716 
3717  /* need to check even locally deleted constraints */
3718  if( andcons == NULL ) /*|| !SCIPconsIsActive(andcons) )*/
3719  continue;
3720 
3721  vars = SCIPgetVarsAnd(scip, andcons);
3722  nvars = SCIPgetNVarsAnd(scip, andcons);
3723  res = SCIPgetResultantAnd(scip, andcons);
3724  assert(nvars == 0 || (vars != NULL && res != NULL));
3725 
3726  andvalue = 1;
3727  /* check if the and-constraint is violated */
3728  for( v = nvars - 1; v >= 0; --v )
3729  {
3730  andvalue *= SCIPgetSolVal(scip, sol, vars[v]);
3731  if( SCIPisFeasZero(scip, andvalue) )
3732  break;
3733  }
3734 
3735  /* check for violation and update aging */
3736  if( !SCIPisFeasEQ(scip, andvalue, SCIPgetSolVal(scip, sol, res)) )
3737  {
3738  /* only reset constraint age if we are in enforcement */
3739  if( sol == NULL )
3740  {
3741  SCIP_CALL( SCIPresetConsAge(scip, andcons) );
3742  }
3743 
3744  *violated = TRUE;
3745  break;
3746  }
3747  else if( sol == NULL )
3748  {
3749  SCIP_CALL( SCIPincConsAge(scip, andcons) );
3750  }
3751  }
3752 
3753  return SCIP_OKAY;
3754 }
3755 
3756 /** creates by copying and captures a linear constraint */
3757 static
3759  SCIP*const targetscip, /**< target SCIP data structure */
3760  SCIP_CONS** targetcons, /**< pointer to store the created target constraint */
3761  SCIP*const sourcescip, /**< source SCIP data structure */
3762  SCIP_CONS*const sourcecons, /**< source constraint which will be copied */
3763  const char* name, /**< name of constraint */
3764  SCIP_HASHMAP*const varmap, /**< a SCIP_HASHMAP mapping variables of the source SCIP to corresponding
3765  * variables of the target SCIP */
3766  SCIP_HASHMAP*const consmap, /**< a hashmap to store the mapping of source constraints to the corresponding
3767  * target constraints */
3768  SCIP_Bool const initial, /**< should the LP relaxation of constraint be in the initial LP? */
3769  SCIP_Bool const separate, /**< should the constraint be separated during LP processing? */
3770  SCIP_Bool const enforce, /**< should the constraint be enforced during node processing? */
3771  SCIP_Bool const check, /**< should the constraint be checked for feasibility? */
3772  SCIP_Bool const propagate, /**< should the constraint be propagated during node processing? */
3773  SCIP_Bool const local, /**< is constraint only valid locally? */
3774  SCIP_Bool const modifiable, /**< is constraint modifiable (subject to column generation)? */
3775  SCIP_Bool const dynamic, /**< is constraint subject to aging? */
3776  SCIP_Bool const removable, /**< should the relaxation be removed from the LP due to aging or cleanup? */
3777  SCIP_Bool const stickingatnode, /**< should the constraint always be kept at the node where it was added, even
3778  * if it may be moved to a more global node? */
3779  SCIP_Bool const global, /**< create a global or a local copy? */
3780  SCIP_Bool*const valid /**< pointer to store if the copying was valid */
3781  )
3782 {
3783  SCIP_CONSDATA* sourceconsdata;
3784  SCIP_CONS* sourcelincons;
3785 
3786  assert(targetscip != NULL);
3787  assert(targetcons != NULL);
3788  assert(sourcescip != NULL);
3789  assert(sourcecons != NULL);
3790  assert(strcmp(SCIPconshdlrGetName(SCIPconsGetHdlr(sourcecons)), CONSHDLR_NAME) == 0);
3791  assert(valid != NULL);
3792 
3793  *valid = TRUE;
3794 
3795  sourceconsdata = SCIPconsGetData(sourcecons);
3796  assert(sourceconsdata != NULL);
3797 
3798  /* get linear constraint */
3799  sourcelincons = sourceconsdata->lincons;
3800  assert(sourcelincons != NULL);
3801 
3802  /* get copied version of linear constraint */
3803  if( !SCIPconsIsDeleted(sourcelincons) )
3804  {
3805  SCIP_CONSHDLR* conshdlrlinear;
3806  SCIP_CONS* targetlincons;
3807  SCIP_CONS** targetandconss;
3808  SCIP_Real* targetandcoefs;
3809  int ntargetandconss;
3810  SCIP_LINEARCONSTYPE targetlinconstype;
3811 
3812  targetlinconstype = sourceconsdata->linconstype;
3813 
3814  switch( targetlinconstype )
3815  {
3817  conshdlrlinear = SCIPfindConshdlr(sourcescip, "linear");
3818  assert(conshdlrlinear != NULL);
3819  break;
3821  conshdlrlinear = SCIPfindConshdlr(sourcescip, "logicor");
3822  assert(conshdlrlinear != NULL);
3823  break;
3825  conshdlrlinear = SCIPfindConshdlr(sourcescip, "knapsack");
3826  assert(conshdlrlinear != NULL);
3827  break;
3829  conshdlrlinear = SCIPfindConshdlr(sourcescip, "setppc");
3830  assert(conshdlrlinear != NULL);
3831  break;
3832 #ifdef WITHEQKNAPSACK
3833  case SCIP_LINEARCONSTYPE_EQKNAPSACK:
3834  conshdlrlinear = SCIPfindConshdlr(sourcescip, "eqknapsack");
3835  assert(conshdlrlinear != NULL);
3836  break;
3837 #endif
3839  default:
3840  SCIPerrorMessage("unknown linear constraint type\n");
3841  return SCIP_INVALIDDATA;
3842  }
3843 
3844  if( conshdlrlinear == NULL ) /*lint !e774*/
3845  {
3846  SCIPerrorMessage("linear constraint handler not found\n");
3847  return SCIP_INVALIDDATA;
3848  }
3849 
3850  targetlincons = NULL;
3851 
3852  /* copy linear constraint */
3853  SCIP_CALL( SCIPgetConsCopy(sourcescip, targetscip, sourcelincons, &targetlincons, conshdlrlinear, varmap, consmap, SCIPconsGetName(sourcelincons),
3854  SCIPconsIsInitial(sourcelincons), SCIPconsIsSeparated(sourcelincons), SCIPconsIsEnforced(sourcelincons), SCIPconsIsChecked(sourcelincons),
3855  SCIPconsIsPropagated(sourcelincons), SCIPconsIsLocal(sourcelincons), SCIPconsIsModifiable(sourcelincons), SCIPconsIsDynamic(sourcelincons),
3856  SCIPconsIsRemovable(sourcelincons), SCIPconsIsStickingAtNode(sourcelincons), global, valid) );
3857 
3858  if( *valid )
3859  {
3860  assert(targetlincons != NULL);
3861  assert(SCIPconsGetHdlr(targetlincons) != NULL);
3862  /* @note due to copying special linear constraints, now leads only to simple linear constraints, we check that
3863  * our target constraint handler is the same as our source constraint handler of the linear constraint,
3864  * if not copying was not valid
3865  */
3866  if( strcmp(SCIPconshdlrGetName(SCIPconsGetHdlr(targetlincons)), "linear") == 0 )
3867  targetlinconstype = SCIP_LINEARCONSTYPE_LINEAR;
3868  }
3869 
3870  targetandconss = NULL;
3871  targetandcoefs = NULL;
3872  ntargetandconss = 0;
3873 
3874  if( *valid )
3875  {
3876  SCIP_CONSHDLR* conshdlrand;
3877  int c;
3878  int nsourceandconss;
3879  SCIP_HASHTABLE* linconsvarsmap;
3880  SCIP_VAR** targetlinvars;
3881  SCIP_Real* targetlincoefs;
3882  int ntargetlinvars;
3883 
3884  conshdlrand = SCIPfindConshdlr(sourcescip, "and");
3885  assert(conshdlrand != NULL);
3886 
3887  nsourceandconss = sourceconsdata->nconsanddatas;
3888 
3889  /* allocate temporary memory */
3890  SCIP_CALL( SCIPallocBufferArray(sourcescip, &targetandconss, nsourceandconss) );
3891  SCIP_CALL( SCIPallocBufferArray(sourcescip, &targetandcoefs, nsourceandconss) );
3892 
3893  /* get the number of vars in the copied linear constraint and allocate buffers
3894  * for the variables and the coefficients
3895  */
3896  SCIP_CALL( getLinearConsNVars(targetscip, targetlincons, targetlinconstype, &ntargetlinvars) );
3897  SCIP_CALL( SCIPallocBufferArray(sourcescip, &targetlinvars, ntargetlinvars) );
3898  SCIP_CALL( SCIPallocBufferArray(sourcescip, &targetlincoefs, ntargetlinvars) );
3899 
3900  /* retrieve the variables of the copied linear constraint */
3901  SCIP_CALL( getLinearConsVarsData(targetscip, targetlincons, targetlinconstype,
3902  targetlinvars, targetlincoefs, &ntargetlinvars) );
3903 
3904  /* now create a hashtable and insert the variables into it, so that it
3905  * can be checked in constant time if a variable was removed due to
3906  * compressed copying when looping over the and resultants
3907  */
3908  SCIP_CALL( SCIPhashtableCreate(&linconsvarsmap, SCIPblkmem(targetscip), ntargetlinvars, SCIPvarGetHashkey,
3909  SCIPvarIsHashkeyEq, SCIPvarGetHashkeyVal, NULL) );
3910 
3911  for( c = 0 ; c < ntargetlinvars; ++c )
3912  {
3913  SCIP_CALL( SCIPhashtableInsert(linconsvarsmap, targetlinvars[c]) );
3914  }
3915 
3916  /* free the buffer arrays that were only required for building the hastable */
3917  SCIPfreeBufferArray(sourcescip, &targetlincoefs);
3918  SCIPfreeBufferArray(sourcescip, &targetlinvars);
3919 
3920  for( c = 0 ; c < nsourceandconss; ++c )
3921  {
3922  CONSANDDATA* consanddata;
3923  SCIP_CONS* oldcons;
3924  SCIP_VAR* targetandresultant;
3925  SCIP_Bool validand;
3926 
3927  consanddata = sourceconsdata->consanddatas[c];
3928  assert(consanddata != NULL);
3929 
3930  oldcons = consanddata->cons;
3931  assert(oldcons != NULL);
3932 
3933  targetandresultant = (SCIP_VAR*) SCIPhashmapGetImage(varmap, SCIPgetResultantAnd(sourcescip, oldcons));
3934  assert(targetandresultant != NULL);
3935 
3936  /* if compressed copying is active, the resultant might not have been copied by the linear
3937  * constraint and we don't need to add it to the pseudo boolean constraint in this case
3938  */
3939  if( !SCIPhashtableExists(linconsvarsmap, targetandresultant) )
3940  continue;
3941 
3942  validand = TRUE;
3943 
3944  targetandconss[ntargetandconss] = NULL;
3945 
3946  /* copy and-constraints */
3947  SCIP_CALL( SCIPgetConsCopy(sourcescip, targetscip, oldcons, &targetandconss[ntargetandconss], conshdlrand, varmap, consmap, SCIPconsGetName(oldcons),
3948  SCIPconsIsInitial(oldcons), SCIPconsIsSeparated(oldcons), SCIPconsIsEnforced(oldcons), SCIPconsIsChecked(oldcons),
3949  SCIPconsIsPropagated(oldcons), SCIPconsIsLocal(oldcons), SCIPconsIsModifiable(oldcons), SCIPconsIsDynamic(oldcons),
3950  SCIPconsIsRemovable(oldcons), SCIPconsIsStickingAtNode(oldcons), global, &validand) );
3951 
3952  *valid &= validand;
3953 
3954  if( validand )
3955  {
3956  targetandcoefs[ntargetandconss] = sourceconsdata->andcoefs[c];
3957  ++ntargetandconss;
3958  }
3959  }
3960 
3961  SCIPhashtableFree(&linconsvarsmap);
3962  assert(ntargetandconss <= ntargetlinvars);
3963  }
3964 
3965  /* no correct pseudoboolean constraint */
3966  if( ntargetandconss == 0 )
3967  {
3968  SCIPdebugMsg(sourcescip, "no and-constraints copied for pseudoboolean constraint <%s>\n", SCIPconsGetName(sourcecons));
3969  *valid = FALSE;
3970  }
3971 
3972  if( *valid )
3973  {
3974  SCIP_Real targetrhs;
3975  SCIP_Real targetlhs;
3976 
3977  SCIP_VAR* intvar;
3978  SCIP_VAR* indvar;
3979  const char* consname;
3980 
3981  /* third the indicator and artificial integer variable part */
3982  assert(sourceconsdata->issoftcons == (sourceconsdata->indvar != NULL));
3983  indvar = sourceconsdata->indvar;
3984  intvar = sourceconsdata->intvar;
3985 
3986  /* copy indicator variable */
3987  if( indvar != NULL )
3988  {
3989  assert(*valid);
3990  SCIP_CALL( SCIPgetVarCopy(sourcescip, targetscip, indvar, &indvar, varmap, consmap, global, valid) );
3991  assert(!(*valid) || indvar != NULL);
3992  }
3993  /* copy artificial integer variable */
3994  if( intvar != NULL && *valid )
3995  {
3996  SCIP_CALL( SCIPgetVarCopy(sourcescip, targetscip, intvar, &intvar, varmap, consmap, global, valid) );
3997  assert(!(*valid) || intvar != NULL);
3998  }
3999 
4000  if( *valid )
4001  {
4002  if( name != NULL )
4003  consname = name;
4004  else
4005  consname = SCIPconsGetName(sourcecons);
4006 
4007  /* get new left and right hand sides of copied linear constraint since
4008  * they might have changed if compressed copying is used
4009  */
4010  SCIP_CALL( getLinearConsSides(targetscip, targetlincons, targetlinconstype, &targetlhs, &targetrhs) );
4011 
4012  /* create new pseudoboolean constraint */
4013  /* coverity[var_deref_op] */
4014  /* coverity[var_deref_model] */
4015  SCIP_CALL( SCIPcreateConsPseudobooleanWithConss(targetscip, targetcons, consname,
4016  targetlincons, targetlinconstype, targetandconss, targetandcoefs, ntargetandconss,
4017  indvar, sourceconsdata->weight, sourceconsdata->issoftcons, intvar, targetlhs, targetrhs,
4018  initial, separate, enforce, check, propagate, local, modifiable, dynamic, removable, stickingatnode) );
4019  }
4020  }
4021 
4022  if( !(*valid) && !SCIPisConsCompressionEnabled(sourcescip) )
4023  {
4024  SCIPverbMessage(sourcescip, SCIP_VERBLEVEL_MINIMAL, NULL, "could not copy constraint <%s>\n", SCIPconsGetName(sourcecons));
4025  }
4026 
4027  /* release copied linear constraint */
4028  if( targetlincons != NULL )
4029  {
4030  SCIP_CALL( SCIPreleaseCons(targetscip, &targetlincons) );
4031  }
4032 
4033  /* release copied and constraint */
4034  if( targetandconss != NULL )
4035  {
4036  int c;
4037 
4038  assert(ntargetandconss <= sourceconsdata->nconsanddatas);
4039 
4040  for( c = 0 ; c < ntargetandconss; ++c )
4041  {
4042  if( targetandconss[c] != NULL )
4043  {
4044  SCIP_CALL( SCIPreleaseCons(targetscip, &targetandconss[c]) );
4045  }
4046  }
4047  }
4048 
4049  /* free temporary memory */
4050  SCIPfreeBufferArrayNull(sourcescip, &targetandcoefs);
4051  SCIPfreeBufferArrayNull(sourcescip, &targetandconss);
4052  }
4053  else
4054  *valid = FALSE;
4055 
4056  return SCIP_OKAY;
4057 }
4058 
4059 /** compute all changes in consanddatas array */
4060 static
4062  SCIP*const scip, /**< SCIP data structure */
4063  SCIP_CONSHDLRDATA*const conshdlrdata /**< pseudoboolean constraint handler data */
4064  )
4065 {
4066  CONSANDDATA** allconsanddatas;
4067  CONSANDDATA* consanddata;
4068  int c;
4069 
4070  assert(scip != NULL);
4071  assert(conshdlrdata != NULL);
4072 
4073  allconsanddatas = conshdlrdata->allconsanddatas;
4074  assert(allconsanddatas != NULL);
4075  assert(conshdlrdata->nallconsanddatas > 0);
4076  assert(conshdlrdata->nallconsanddatas <= conshdlrdata->sallconsanddatas);
4077 
4078  for( c = conshdlrdata->nallconsanddatas - 1; c >= 0; --c )
4079  {
4080  SCIP_CONS* cons;
4081  SCIP_VAR** vars;
4082  int nvars;
4083  SCIP_VAR** newvars;
4084  int nnewvars;
4085  int v;
4086 
4087  consanddata = allconsanddatas[c];
4088 
4089  if( !consanddata->istransformed )
4090  continue;
4091 
4092  if( consanddata->nuses == 0 )
4093  continue;
4094 
4095  vars = consanddata->vars;
4096  nvars = consanddata->nvars;
4097  assert(nvars == 0 || vars != NULL);
4098  assert(consanddata->nnewvars == 0 && ((consanddata->snewvars > 0) == (consanddata->newvars != NULL)));
4099 
4100  if( nvars == 0 )
4101  {
4102 #ifndef NDEBUG
4103  /* if an old consanddata-object has no variables left there should be no new variables */
4104  if( consanddata->cons != NULL )
4105  assert(SCIPgetNVarsAnd(scip, consanddata->cons) == 0);
4106 #endif
4107  continue;
4108  }
4109 
4110  cons = consanddata->cons;
4111  assert(cons != NULL);
4112 
4113  if( SCIPconsIsDeleted(cons) )
4114  continue;
4115 
4116  /* sort and-variables */
4117  if( !SCIPisAndConsSorted(scip, consanddata->cons) )
4118  {
4119  SCIP_CALL( SCIPsortAndCons(scip, consanddata->cons) );
4120  assert(SCIPisAndConsSorted(scip, consanddata->cons));
4121  }
4122 
4123  /* get new and-variables */
4124  nnewvars = SCIPgetNVarsAnd(scip, consanddata->cons);
4125  newvars = SCIPgetVarsAnd(scip, consanddata->cons);
4126 
4127  /* stop if the constraint has no variables or there was an error (coverity issue) */
4128  if( nnewvars <= 0 )
4129  continue;
4130 
4131 #ifndef NDEBUG
4132  /* check that old variables are sorted */
4133  for( v = nvars - 1; v > 0; --v )
4134  assert(SCIPvarGetIndex(vars[v]) >= SCIPvarGetIndex(vars[v - 1]));
4135  /* check that new variables are sorted */
4136  for( v = nnewvars - 1; v > 0; --v )
4137  assert(SCIPvarGetIndex(newvars[v]) >= SCIPvarGetIndex(newvars[v - 1]));
4138 #endif
4139 
4140  /* check for changings, if and-constraint did not change we do not need to copy all variables */
4141  if( nvars == nnewvars )
4142  {
4143  SCIP_Bool changed;
4144 
4145  changed = FALSE;
4146 
4147  /* check each variable */
4148  for( v = nvars - 1; v >= 0; --v )
4149  {
4150  if( vars[v] != newvars[v] )
4151  {
4152  changed = TRUE;
4153  break;
4154  }
4155  }
4156 
4157  if( !changed )
4158  continue;
4159  }
4160 
4161  /* resize newvars array if necessary */
4162  if( nnewvars > consanddata->snewvars )
4163  {
4164  SCIP_CALL( SCIPensureBlockMemoryArray(scip, &(consanddata->newvars), &(consanddata->snewvars), nnewvars) );
4165  }
4166 
4167  /* copy all variables */
4168  BMScopyMemoryArray(consanddata->newvars, newvars, nnewvars);
4169  consanddata->nnewvars = nnewvars;
4170 
4171  /* capture all variables */
4172  for( v = consanddata->nnewvars - 1; v >= 0; --v )
4173  {
4174  /* in original problem the variables was already deleted */
4175  assert(consanddata->newvars[v] != NULL);
4176  SCIP_CALL( SCIPcaptureVar(scip, consanddata->newvars[v]) );
4177  }
4178  }
4179 
4180  return SCIP_OKAY;
4181 }
4182 
4183 /** remove old locks */
4184 static
4186  SCIP*const scip, /**< SCIP data structure */
4187  SCIP_CONS*const cons, /**< pseudoboolean constraint */
4188  CONSANDDATA*const consanddata, /**< CONSANDDATA object for which we want to delete the locks and the
4189  * capture of the corresponding and-constraint */
4190  SCIP_Real const coef, /**< coefficient which led to old locks */
4191  SCIP_Real const lhs, /**< left hand side which led to old locks */
4192  SCIP_Real const rhs /**< right hand side which led to old locks */
4193  )
4194 {
4195  assert(scip != NULL);
4196  assert(cons != NULL);
4197  assert(consanddata != NULL);
4198  assert(!SCIPisInfinity(scip, coef) && !SCIPisInfinity(scip, -coef));
4199  assert(!SCIPisInfinity(scip, lhs));
4200  assert(!SCIPisInfinity(scip, -rhs));
4201  assert(SCIPisLE(scip, lhs, rhs));
4202 
4203  /* remove rounding locks */
4204  SCIP_CALL( unlockRoundingAndCons(scip, cons, consanddata, coef, lhs, rhs) );
4205 
4206  assert(consanddata->cons != NULL);
4207 
4208  return SCIP_OKAY;
4209 }
4210 
4211 /** add new locks */
4212 static
4214  SCIP*const scip, /**< SCIP data structure */
4215  SCIP_CONS*const cons, /**< pseudoboolean constraint */
4216  CONSANDDATA*const consanddata, /**< CONSANDDATA object for which we want to delete the locks and the
4217  * capture of the corresponding and-constraint */
4218  SCIP_Real const coef, /**< coefficient which lead to new locks */
4219  SCIP_Real const lhs, /**< left hand side which lead to new locks */
4220  SCIP_Real const rhs /**< right hand side which lead to new locks */
4221  )
4222 {
4223  assert(scip != NULL);
4224  assert(cons != NULL);
4225  assert(consanddata != NULL);
4226  assert(!SCIPisInfinity(scip, coef) && !SCIPisInfinity(scip, -coef));
4227  assert(!SCIPisInfinity(scip, lhs));
4228  assert(!SCIPisInfinity(scip, -rhs));
4229  assert(SCIPisLE(scip, lhs, rhs));
4230 
4231  /* add rounding locks due to old variables in consanddata object */
4232  SCIP_CALL( lockRoundingAndCons(scip, cons, consanddata, coef, lhs, rhs) );
4233 
4234  assert(consanddata->cons != NULL);
4235 
4236  return SCIP_OKAY;
4237 }
4238 
4239 /** update all locks inside this constraint and all captures on all and-constraints */
4240 static
4242  SCIP*const scip, /**< SCIP data structure */
4243  SCIP_CONS*const cons, /**< pseudoboolean constraint */
4244  SCIP_CONSHDLRDATA*const conshdlrdata, /**< pseudoboolean constraint handler data */
4245  SCIP_Real const newlhs, /**< new left hand side of pseudoboolean constraint */
4246  SCIP_Real const newrhs, /**< new right hand side of pseudoboolean constraint */
4247  SCIP_VAR**const andress, /**< current and-resultants in pseudoboolean constraint */
4248  SCIP_Real*const andcoefs, /**< current and-resultants-coeffcients in pseudoboolean constraint */
4249  SCIP_Bool*const andnegs, /**< current negation status of and-resultants in pseudoboolean constraint */
4250  int const nandress /**< number of current and-resultants in pseudoboolean constraint */
4251  )
4252 {
4253  CONSANDDATA** newconsanddatas;
4254  int nnewconsanddatas;
4255  int snewconsanddatas;
4256  SCIP_Real* newandcoefs;
4257  SCIP_Real* oldandcoefs;
4258  SCIP_Bool* newandnegs;
4259  SCIP_Bool* oldandnegs;
4260  CONSANDDATA** consanddatas;
4261  int nconsanddatas;
4262  SCIP_CONSDATA* consdata;
4263  int oldnvars;
4264  int c;
4265  int c1;
4266 
4267  assert(scip != NULL);
4268  assert(cons != NULL);
4269  assert(conshdlrdata != NULL);
4270  assert(conshdlrdata->hashmap != NULL);
4271  assert(nandress == 0 || (andress != NULL && andcoefs != NULL));
4272  assert(!SCIPisInfinity(scip, newlhs));
4273  assert(!SCIPisInfinity(scip, -newrhs));
4274  assert(SCIPisLE(scip, newlhs, newrhs));
4275 
4276  consdata = SCIPconsGetData(cons);
4277  assert(consdata != NULL);
4278 
4279  /* sort and-constraints after indices of corresponding and-resultants */
4280  SCIPsortPtrRealBool((void**)(consdata->consanddatas), consdata->andcoefs, consdata->andnegs, resvarCompWithInactive, consdata->nconsanddatas);
4281 
4282  consanddatas = consdata->consanddatas;
4283  oldandcoefs = consdata->andcoefs;
4284  oldandnegs = consdata->andnegs;
4285  nconsanddatas = consdata->nconsanddatas;
4286  assert(nconsanddatas == 0 || (consanddatas != NULL && oldandcoefs != NULL));
4287 
4288 #ifndef NDEBUG
4289  /* check that and-resultants are sorted, and coefficents are not zero */
4290  for( c = nandress - 1; c > 0; --c )
4291  {
4292  assert(!SCIPisZero(scip, andcoefs[c]));
4293  assert(SCIPvarGetIndex(andress[c]) > SCIPvarGetIndex(andress[c - 1]));
4294  }
4295  /* check that consanddata objects are sorted due to the index of the corresponding resultants, and coefficents are
4296  * not zero
4297  */
4298  for( c = nconsanddatas - 1; c > 0; --c )
4299  {
4300  SCIP_VAR* res1;
4301  SCIP_VAR* res2;
4302 
4303  assert(consanddatas[c] != NULL);
4304 
4305  if( !consanddatas[c]->istransformed )
4306  continue;
4307 
4308  assert(!SCIPisZero(scip, oldandcoefs[c]));
4309  assert(consanddatas[c - 1] != NULL);
4310 
4311  if( !consanddatas[c - 1]->istransformed )
4312  continue;
4313 
4314  assert(!SCIPisZero(scip, oldandcoefs[c - 1]));
4315 
4316  if( SCIPconsIsDeleted(consanddatas[c]->cons) || SCIPconsIsDeleted(consanddatas[c - 1]->cons) )
4317  continue;
4318 
4319  assert(consanddatas[c]->cons != NULL);
4320  res1 = SCIPgetResultantAnd(scip, consanddatas[c]->cons);
4321  assert(res1 != NULL);
4322  assert(consanddatas[c - 1]->cons != NULL);
4323  res2 = SCIPgetResultantAnd(scip, consanddatas[c - 1]->cons);
4324  assert(res2 != NULL);
4325 
4326  assert(SCIPvarGetIndex(res1) >= SCIPvarGetIndex(res2));
4327  }
4328 #endif
4329 
4330  snewconsanddatas = nconsanddatas + nandress;
4331 
4332  /* allocate new block memory arrays */
4333  SCIP_CALL( SCIPallocBlockMemoryArray(scip, &newconsanddatas, snewconsanddatas) );
4334  SCIP_CALL( SCIPallocBlockMemoryArray(scip, &newandcoefs, snewconsanddatas) );
4335  SCIP_CALL( SCIPallocBlockMemoryArray(scip, &newandnegs, snewconsanddatas) );
4336 
4337  nnewconsanddatas = 0;
4338 
4339  /* collect new consanddata objects and update locks and captures */
4340  for( c = 0, c1 = 0; c < nconsanddatas && c1 < nandress; )
4341  {
4342  SCIP_CONS* andcons;
4343  SCIP_VAR* res1;
4344  SCIP_VAR* res2;
4345 
4346  assert(consanddatas[c] != NULL);
4347 
4348  /* consanddata object could have been deleted in the last presolving round */
4349  if( !consanddatas[c]->istransformed )
4350  {
4351  ++c;
4352  consdata->changed = TRUE;
4353  consdata->upgradetried = FALSE;
4354  continue;
4355  }
4356 
4357  andcons = consanddatas[c]->cons;
4358  assert(andcons != NULL);
4359 
4360  if( andcons == NULL ) /*lint !e774*/
4361  {
4362  ++c;
4363  consdata->changed = TRUE;
4364  consdata->upgradetried = FALSE;
4365  continue;
4366  }
4367  else if( SCIPconsIsDeleted(andcons) )
4368  {
4369  /* remove rounding locks, because the and constraint was deleted */
4370  SCIP_CALL( unlockRoundingAndCons(scip, cons, consanddatas[c],
4371  oldandnegs[c] ? -oldandcoefs[c] : oldandcoefs[c], consdata->lhs, consdata->rhs) );
4372  ++c;
4373  consdata->changed = TRUE;
4374  consdata->upgradetried = FALSE;
4375  continue;
4376  }
4377  assert(andcons != NULL);
4378 
4379  /* get and-resultants of consanddata object in constraint data */
4380  res1 = SCIPgetResultantAnd(scip, andcons);
4381  assert(res1 != NULL);
4382  assert(SCIPhashmapGetImage(conshdlrdata->hashmap, (void*)res1) == consanddatas[c]);
4383 
4384  /* get and-resultants in new corresponding linear constraint */
4385  res2 = andress[c1];
4386  assert(res2 != NULL);
4387  assert(SCIPhashmapGetImage(conshdlrdata->hashmap, (void*)res2) != NULL);
4388 
4389  /* collect new consanddata objects in sorted order due to the variable index of corresponding and-resultants */
4390  if( SCIPvarGetIndex(res1) < SCIPvarGetIndex(res2) )
4391  {
4392  assert(consanddatas[c]->nuses > 0);
4393  --(consanddatas[c]->nuses);
4394 
4395  /* remove old locks */
4396  SCIP_CALL( removeOldLocks(scip, cons, consanddatas[c], oldandnegs[c] ? -oldandcoefs[c] : oldandcoefs[c],
4397  consdata->lhs, consdata->rhs) );
4398  ++c;
4399  consdata->changed = TRUE;
4400  consdata->upgradetried = FALSE;
4401  consdata->propagated = FALSE;
4402  consdata->presolved = FALSE;
4403  }
4404  else if( SCIPvarGetIndex(res1) > SCIPvarGetIndex(res2) )
4405  {
4406  assert(SCIPhashmapExists(conshdlrdata->hashmap, (void*)res2));
4407  newconsanddatas[nnewconsanddatas] = (CONSANDDATA*) SCIPhashmapGetImage(conshdlrdata->hashmap, (void*)res2);
4408  newandcoefs[nnewconsanddatas] = andcoefs[c1];
4409  newandnegs[nnewconsanddatas] = andnegs[c1];
4410  ++(newconsanddatas[nnewconsanddatas]->nuses);
4411 
4412  /* add new locks */
4413  SCIP_CALL( addNewLocks(scip, cons, newconsanddatas[nnewconsanddatas], newandnegs[nnewconsanddatas] ?
4414  -newandcoefs[nnewconsanddatas] : newandcoefs[nnewconsanddatas], newlhs, newrhs) );
4415  ++c1;
4416  consdata->changed = TRUE;
4417  consdata->upgradetried = FALSE;
4418  consdata->cliquesadded = FALSE;
4419  consdata->propagated = FALSE;
4420  consdata->presolved = FALSE;
4421 
4422  ++nnewconsanddatas;
4423  }
4424  else
4425  {
4426  SCIP_Bool coefsignchanged;
4427  SCIP_Bool lhschanged;
4428  SCIP_Bool rhschanged;
4429 
4430  assert(SCIPhashmapGetImage(conshdlrdata->hashmap, (void*)res2) == consanddatas[c]);
4431 
4432  /* copy old consanddata object and new coefficent */
4433  newconsanddatas[nnewconsanddatas] = consanddatas[c];
4434 
4435  newandcoefs[nnewconsanddatas] = andcoefs[c1];
4436  newandnegs[nnewconsanddatas] = andnegs[c1];
4437 
4438  if( ((oldandnegs[c] == andnegs[c1]) && !SCIPisEQ(scip, oldandcoefs[c], newandcoefs[c1]))
4439  || ((oldandnegs[c] != newandnegs[c1]) && !SCIPisEQ(scip, oldandcoefs[c], -newandcoefs[c1])) )
4440  consdata->upgradetried = FALSE;
4441 
4442  coefsignchanged = (oldandnegs[c] == andnegs[c1]) &&
4443  ((oldandcoefs[c] < 0 && andcoefs[c1] > 0) || (oldandcoefs[c] > 0 && andcoefs[c1] < 0));
4444  coefsignchanged = coefsignchanged || ((oldandnegs[c] != andnegs[c1]) &&
4445  ((oldandcoefs[c] < 0 && andcoefs[c1] < 0) || (oldandcoefs[c] > 0 && andcoefs[c1] > 0)));
4446  lhschanged = (SCIPisInfinity(scip, -consdata->lhs) && !SCIPisInfinity(scip, -newlhs)) || (!SCIPisInfinity(scip, -consdata->lhs) && SCIPisInfinity(scip, -newlhs))
4447  || (consdata->lhs < 0 && newlhs > 0) || (consdata->lhs > 0 && newlhs < 0);
4448  rhschanged = (SCIPisInfinity(scip, consdata->rhs) && !SCIPisInfinity(scip, newrhs)) || (!SCIPisInfinity(scip, consdata->rhs) && SCIPisInfinity(scip, newrhs))
4449  || (consdata->rhs < 0 && newrhs > 0) || (consdata->rhs > 0 && newrhs < 0);
4450 
4451  /* update or renew locks */
4452  if( coefsignchanged || lhschanged || rhschanged || newconsanddatas[nnewconsanddatas]->nnewvars > 0)
4453  {
4454  /* renew locks */
4455  SCIP_CALL( removeOldLocks(scip, cons, newconsanddatas[nnewconsanddatas], oldandnegs[c] ?
4456  -oldandcoefs[c] : oldandcoefs[c], consdata->lhs, consdata->rhs) );
4457  SCIP_CALL( addNewLocks(scip, cons, newconsanddatas[nnewconsanddatas], newandnegs[nnewconsanddatas] ?
4458  -newandcoefs[nnewconsanddatas] : newandcoefs[nnewconsanddatas], newlhs, newrhs) );
4459 
4460  consdata->changed = TRUE;
4461  consdata->upgradetried = FALSE;
4462  consdata->cliquesadded = FALSE;
4463  consdata->propagated = FALSE;
4464  consdata->presolved = FALSE;
4465  }
4466 
4467  ++c;
4468  ++c1;
4469  ++nnewconsanddatas;
4470  }
4471  }
4472 
4473  /* add all remaining consanddatas and update locks and captures */
4474  if( c < nconsanddatas )
4475  {
4476  assert(c1 == nandress);
4477 
4478  for( ; c < nconsanddatas; ++c )
4479  {
4480  SCIP_CONS* andcons;
4481 #ifndef NDEBUG
4482  SCIP_VAR* res1;
4483 
4484  assert(consanddatas[c] != NULL);
4485 #endif
4486  andcons = consanddatas[c]->cons;
4487 #ifndef NDEBUG
4488  if( andcons != NULL )
4489  {
4490  res1 = SCIPgetResultantAnd(scip, andcons);
4491  assert(res1 != NULL);
4492  assert(SCIPhashmapGetImage(conshdlrdata->hashmap, (void*)res1) == consanddatas[c]);
4493  }
4494 #endif
4495  if( andcons == NULL )
4496  {
4497  consdata->changed = TRUE;
4498  consdata->upgradetried = FALSE;
4499  continue;
4500  }
4501 
4502  assert(consanddatas[c]->nuses > 0);
4503  --(consanddatas[c]->nuses);
4504 
4505  /* remove old locks */
4506  SCIP_CALL( removeOldLocks(scip, cons, consanddatas[c], oldandnegs[c] ? -oldandcoefs[c] : oldandcoefs[c],
4507  consdata->lhs, consdata->rhs) );
4508  consdata->changed = TRUE;
4509  consdata->upgradetried = FALSE;
4510  consdata->propagated = FALSE;
4511  consdata->presolved = FALSE;
4512  }
4513  }
4514  else if( c1 < nandress )
4515  {
4516  for( ; c1 < nandress; ++c1 )
4517  {
4518  SCIP_VAR* res2;
4519 
4520  res2 = andress[c1];
4521  assert(res2 != NULL);
4522  assert(SCIPhashmapExists(conshdlrdata->hashmap, (void*)res2));
4523  newconsanddatas[nnewconsanddatas] = (CONSANDDATA*) SCIPhashmapGetImage(conshdlrdata->hashmap, (void*)res2);
4524  newandcoefs[nnewconsanddatas] = andcoefs[c1];
4525  newandnegs[nnewconsanddatas] = andnegs[c1];
4526  ++(newconsanddatas[nnewconsanddatas]->nuses);
4527 
4528  /* add new locks */
4529  SCIP_CALL( addNewLocks(scip, cons, newconsanddatas[nnewconsanddatas], newandnegs[nnewconsanddatas] ?
4530  -newandcoefs[nnewconsanddatas] : newandcoefs[nnewconsanddatas], newlhs, newrhs) );
4531 
4532  ++nnewconsanddatas;
4533  consdata->changed = TRUE;
4534  consdata->upgradetried = FALSE;
4535  consdata->cliquesadded = FALSE;
4536  consdata->propagated = FALSE;
4537  consdata->presolved = FALSE;
4538  }
4539  }
4540  assert(c == nconsanddatas && c1 == nandress);
4541 
4542  /* delete old and-coefficients and consanddata objects */
4543  SCIPfreeBlockMemoryArray(scip, &(consdata->andcoefs), consdata->sconsanddatas);
4544  SCIPfreeBlockMemoryArray(scip, &(consdata->andnegs), consdata->sconsanddatas);
4545  SCIPfreeBlockMemoryArray(scip, &(consdata->consanddatas), consdata->sconsanddatas);
4546 
4547  if( !SCIPisEQ(scip, consdata->lhs, newlhs) || !SCIPisEQ(scip, consdata->rhs, newrhs) )
4548  {
4549  consdata->upgradetried = FALSE;
4550  consdata->lhs = newlhs;
4551  consdata->rhs = newrhs;
4552  }
4553 
4554  consdata->consanddatas = newconsanddatas;
4555  consdata->andcoefs = newandcoefs;
4556  consdata->andnegs = newandnegs;
4557  consdata->nconsanddatas = nnewconsanddatas;
4558  consdata->sconsanddatas = snewconsanddatas;
4559 
4560  oldnvars = consdata->nlinvars;
4561  /* update number of linear variables without and-resultants */
4562  SCIP_CALL( getLinearConsNVars(scip, consdata->lincons, consdata->linconstype, &(consdata->nlinvars)) );
4563  consdata->nlinvars -= nnewconsanddatas;
4564 
4565  if( oldnvars != consdata->nlinvars )
4566  {
4567  consdata->changed = TRUE;
4568  consdata->upgradetried = FALSE;
4569  consdata->cliquesadded = FALSE;
4570  consdata->propagated = FALSE;
4571  consdata->presolved = FALSE;
4572  }
4573 
4574  /* we need to re-sort and-constraints after indices of corresponding and-resultants, since we might have replaced
4575  * negated variables
4576  */
4577  SCIPsortPtrRealBool((void**)(consdata->consanddatas), consdata->andcoefs, consdata->andnegs, resvarCompWithInactive, consdata->nconsanddatas);
4578 
4579 #ifndef NDEBUG
4580  consanddatas = consdata->consanddatas;
4581  nconsanddatas = consdata->nconsanddatas;
4582  assert(nconsanddatas == 0 || consanddatas != NULL);
4583 
4584  /* check that consanddata objects are sorted with respect to the index of the corresponding resultants */
4585  for( c = nconsanddatas - 1; c > 0; --c )
4586  {
4587  SCIP_VAR* res1;
4588  SCIP_VAR* res2;
4589 
4590  assert(consanddatas[c] != NULL);
4591  assert(consanddatas[c]->cons != NULL);
4592  res1 = SCIPgetResultantAnd(scip, consanddatas[c]->cons);
4593  assert(res1 != NULL);
4594  assert(consanddatas[c - 1] != NULL);
4595  assert(consanddatas[c - 1]->cons != NULL);
4596  res2 = SCIPgetResultantAnd(scip, consanddatas[c - 1]->cons);
4597  assert(res2 != NULL);
4598 
4599  assert(SCIPvarGetIndex(res1) > SCIPvarGetIndex(res2));
4600  }
4601 #endif
4602 
4603  return SCIP_OKAY;
4604 }
4605 
4606 /** adds cliques of the pseudoboolean constraint to the global clique table */
4607 static
4609  SCIP*const scip, /**< SCIP data structure */
4610  SCIP_CONS*const cons, /**< pseudoboolean constraint */
4611  SCIP_Bool*const cutoff, /**< pointer to store whether the node can be cut off */
4612  int*const naggrvars, /**< pointer to count the number of aggregated variables */
4613  int*const nchgbds /**< pointer to count the number of performed bound changes */
4614  )
4615 {
4616  SCIP_CONSDATA* consdata;
4617  SCIP_VAR** vars;
4618  int nvars;
4619  SCIP_VAR** linvars;
4620  SCIP_VAR* andres;
4621  SCIP_VAR* andres2;
4622  int nlinvars;
4623  int nandress;
4624  int c;
4625  int v2;
4626  int v1;
4627  int nchgbdslocal;
4628 
4629  assert(scip != NULL);
4630  assert(cons != NULL);
4631  assert(cutoff != NULL);
4632  assert(naggrvars != NULL);
4633  assert(nchgbds != NULL);
4634  assert(SCIPconsIsActive(cons));
4635 
4636  *cutoff = FALSE;
4637 
4638  consdata = SCIPconsGetData(cons);
4639  assert(consdata != NULL);
4640  /* if we have no and-constraints left, we should not be here and this constraint should be deleted (only the linaer should survive) */
4641  assert(consdata->nconsanddatas > 0);
4642 
4643  /* check whether the cliques have already been added */
4644  if( consdata->cliquesadded )
4645  return SCIP_OKAY;
4646 
4647  consdata->cliquesadded = TRUE;
4648 
4649  checkConsConsistency(scip, cons);
4650 
4651  /* check standard pointers and sizes */
4652  assert(consdata->lincons != NULL);
4653  assert(SCIPconsIsActive(consdata->lincons));
4654  assert(consdata->linconstype > SCIP_LINEARCONSTYPE_INVALIDCONS);
4655  assert(consdata->consanddatas != NULL);
4656  assert(consdata->nconsanddatas > 0);
4657  assert(consdata->nconsanddatas <= consdata->sconsanddatas);
4658 
4659  /* check number of linear variables */
4660  SCIP_CALL( getLinearConsNVars(scip, consdata->lincons, consdata->linconstype, &nvars) );
4661  assert(nvars == consdata->nlinvars + consdata->nconsanddatas);
4662 
4663  /* get temporary memory */
4664  SCIP_CALL( SCIPallocBufferArray(scip, &vars, nvars) );
4665  SCIP_CALL( SCIPallocBufferArray(scip, &linvars, nvars) );
4666 
4667  /* get variables and coefficients */
4668  SCIP_CALL( getLinearConsVarsData(scip, consdata->lincons, consdata->linconstype, vars, NULL, &nvars) );
4669 
4670  /* calculate all not artificial linear variables and all artificial and-resultants which will be ordered like the
4671  * 'consanddatas' such that the and-resultant of the and-constraint is the and-resultant in the 'andress' array
4672  * afterwards
4673  * @todo should we take into accout the negation status of the cliques?
4674  */
4675  SCIP_CALL( getLinVarsAndAndRess(scip, cons, vars, NULL, nvars, linvars, NULL, &nlinvars,
4676  NULL, NULL, NULL, &nandress) );
4677 
4678  assert(nandress == consdata->nconsanddatas);
4679  assert(consdata->consanddatas != NULL);
4680 
4681  /* find cliques from linear variable to and-resultant */
4682  for( c = nandress - 1; c >= 0; --c )
4683  {
4684  CONSANDDATA* consanddata;
4685  SCIP_VAR** andvars;
4686  int nandvars;
4687 
4688  consanddata = consdata->consanddatas[c];
4689  assert(consanddata != NULL);
4690 
4691  andres = SCIPgetResultantAnd(scip, consanddata->cons);
4692 
4693  /* choose correct variable array */
4694  if( consanddata->nnewvars > 0 )
4695  {
4696  andvars = consanddata->newvars;
4697  nandvars = consanddata->nnewvars;
4698  }
4699  else
4700  {
4701  andvars = consanddata->vars;
4702  nandvars = consanddata->nvars;
4703  }
4704 
4705  for( v1 = nandvars - 1; v1 >= 0; --v1 )
4706  {
4707  SCIP_VAR* var1;
4708  SCIP_Bool values[2];
4709 
4710  var1 = andvars[v1];
4711  if( !SCIPvarIsActive(var1) && (!SCIPvarIsNegated(var1) || !SCIPvarIsActive(SCIPvarGetNegationVar(var1))) )
4712  continue;
4713 
4714  /* get active counterpart to check for common cliques */
4716  {
4717  var1 = SCIPvarGetNegationVar(var1);
4718  values[0] = FALSE;
4719  }
4720  else
4721  values[0] = TRUE;
4722 
4723  for( v2 = nlinvars - 1; v2 >= 0; --v2 )
4724  {
4725  SCIP_VAR* var2;
4726 
4727  var2 = linvars[v2];
4728  if( !SCIPvarIsActive(var2) && (!SCIPvarIsNegated(var2) || !SCIPvarIsActive(SCIPvarGetNegationVar(var2))) )
4729  continue;
4730 
4731  /* get active counterpart to check for common cliques */
4733  {
4734  var2 = SCIPvarGetNegationVar(var2);
4735  values[1] = FALSE;
4736  }
4737  else
4738  values[1] = TRUE;
4739 
4740  /* if variable in and-constraint1 is the negated variable of a normal linear variable, than we can add a
4741  * clique between the and-resultant and the normal linear variable, negated variables are not save in
4742  * cliquetables
4743  *
4744  * set r_1 = var1 * z; (z is some product)
4745  * var1 == ~var2
4746  *
4747  * if:
4748  * var1 + ~var1 <= 1; r_1
4749  * 0 + 1 <= 1 0 \
4750  * 1 + 0 <= 1 ==> 1 or 0 > ==> r_1 + var2 <= 1
4751  * 0 + 0 <= 1 0 /
4752  */
4753  if( values[0] != values[1] && var1 == var2 )
4754  {
4755  SCIP_CONS* newcons;
4756  SCIP_VAR* clqvars[2];
4757  char consname[SCIP_MAXSTRLEN];
4758 
4759  clqvars[0] = andres;
4760  clqvars[1] = values[1] ? var2 : SCIPvarGetNegatedVar(var2);
4761  assert(clqvars[1] != NULL);
4762 
4763  /* @todo: check whether it is better to only add the clique or to add the setppc constraint or do both */
4764 
4765  /* add clique */
4766  SCIP_CALL( SCIPaddClique(scip, clqvars, NULL, 2, FALSE, cutoff, &nchgbdslocal) );
4767  if( *cutoff )
4768  goto TERMINATE;
4769 
4770  *nchgbds += nchgbdslocal;
4771 
4772  (void) SCIPsnprintf(consname, SCIP_MAXSTRLEN, "%s_clq_%s_%s", SCIPconsGetName(cons), SCIPvarGetName(clqvars[0]), SCIPvarGetName(clqvars[1]) );
4773  SCIP_CALL( SCIPcreateConsSetpack(scip, &newcons, consname, 2, clqvars,
4775  FALSE, SCIPconsIsPropagated(cons),
4778 
4779  SCIP_CALL( SCIPaddCons(scip, newcons) );
4780  SCIPdebugMsg(scip, "added a clique/setppc constraint <%s> \n", SCIPconsGetName(newcons));
4781  SCIPdebugPrintCons(scip, newcons, NULL);
4782 
4783  SCIP_CALL( SCIPreleaseCons(scip, &newcons) );
4784  }
4785  /* if a variable in an and-constraint is in a clique with another normal linear variable, we can add the
4786  * clique between the linear variable and the and-resultant
4787  *
4788  * set r_1 = var1 * z; (z is some product)
4789  *
4790  * if:
4791  * var1 + var2 <= 1; r_1
4792  * 0 + 1 <= 1 0 \
4793  * 1 + 0 <= 1 ==> 1 or 0 > ==> r_1 + var2 <= 1
4794  * 0 + 0 <= 1 0 /
4795  */
4796  if( (var1 != var2) && SCIPvarsHaveCommonClique(var1, values[0], var2, values[1], TRUE) )
4797  {
4798  SCIP_CONS* newcons;
4799  SCIP_VAR* clqvars[2];
4800  char consname[SCIP_MAXSTRLEN];
4801 
4802  clqvars[0] = andres;
4803  clqvars[1] = values[1] ? var2 : SCIPvarGetNegatedVar(var2);
4804  assert(clqvars[1] != NULL);
4805 
4806  /* @todo: check whether it is better to only add the clique or to add the setppc constraint or do both */
4807 
4808  /* add clique */
4809  SCIP_CALL( SCIPaddClique(scip, clqvars, NULL, 2, FALSE, cutoff, &nchgbdslocal) );
4810  if( *cutoff )
4811  goto TERMINATE;
4812 
4813  *nchgbds += nchgbdslocal;
4814 
4815  (void) SCIPsnprintf(consname, SCIP_MAXSTRLEN, "%s_clq_%s_%s", SCIPconsGetName(cons), SCIPvarGetName(clqvars[0]), SCIPvarGetName(clqvars[1]) );
4816  SCIP_CALL( SCIPcreateConsSetpack(scip, &newcons, consname, 2, clqvars,
4818  FALSE, SCIPconsIsPropagated(cons),
4821 
4822  SCIP_CALL( SCIPaddCons(scip, newcons) );
4823  SCIPdebugMsg(scip, "added a clique/setppc constraint <%s> \n", SCIPconsGetName(newcons));
4824  SCIPdebugPrintCons(scip, newcons, NULL);
4825 
4826  SCIP_CALL( SCIPreleaseCons(scip, &newcons) );
4827  }
4828  }
4829  }
4830  }
4831 
4832  /* find cliques over variables which are in different and-constraints */
4833  for( c = nandress - 1; c > 0; --c )
4834  {
4835  CONSANDDATA* consanddata1;
4836  CONSANDDATA* consanddata2;
4837  SCIP_VAR** andvars1;
4838  int nandvars1;
4839  SCIP_VAR** andvars2;
4840  int nandvars2;
4841 
4842  consanddata1 = consdata->consanddatas[c];
4843  assert(consanddata1 != NULL);
4844  consanddata2 = consdata->consanddatas[c - 1];
4845  assert(consanddata2 != NULL);
4846 
4847  andres = SCIPgetResultantAnd(scip, consanddata1->cons);
4848  andres2 = SCIPgetResultantAnd(scip, consanddata2->cons);
4849 
4850  /* choose correct variable array of consanddata object 1 */
4851  if( consanddata1->nnewvars > 0 )
4852  {
4853  andvars1 = consanddata1->newvars;
4854  nandvars1 = consanddata1->nnewvars;
4855  }
4856  else
4857  {
4858  andvars1 = consanddata1->vars;
4859  nandvars1 = consanddata1->nvars;
4860  }
4861 
4862  /* choose correct variable array of consanddata object 2 */
4863  if( consanddata2->nnewvars > 0 )
4864  {
4865  andvars2 = consanddata2->newvars;
4866  nandvars2 = consanddata2->nnewvars;
4867  }
4868  else
4869  {
4870  andvars2 = consanddata2->vars;
4871  nandvars2 = consanddata2->nvars;
4872  }
4873 
4874  /* compare both terms for finding new aggregated variables and new cliques */
4875  for( v1 = nandvars1 - 1; v1 >= 0; --v1 )
4876  {
4877  SCIP_VAR* var1;
4878  SCIP_Bool values[2];
4879 
4880  var1 = andvars1[v1];
4881  if( !SCIPvarIsActive(var1) && (!SCIPvarIsNegated(var1) || !SCIPvarIsActive(SCIPvarGetNegationVar(var1))) )
4882  continue;
4883 
4884  /* get active counterpart to check for common cliques */
4886  {
4887  var1 = SCIPvarGetNegationVar(var1);
4888  values[0] = FALSE;
4889  }
4890  else
4891  values[0] = TRUE;
4892 
4893  for( v2 = nandvars2 - 1; v2 >= 0; --v2 )
4894  {
4895  SCIP_VAR* var2;
4896 
4897  var2 = andvars2[v2];
4898  if( !SCIPvarIsActive(var2) && (!SCIPvarIsNegated(var2) || !SCIPvarIsActive(SCIPvarGetNegationVar(var2))) )
4899  continue;
4900 
4901  /* get active counterpart to check for common cliques */
4903  {
4904  var2 = SCIPvarGetNegationVar(var2);
4905  values[1] = FALSE;
4906  }
4907  else
4908  values[1] = TRUE;
4909 
4910  /* if a variable in and-constraint1 is the negated variable of a variable in and-constraint2, than we can
4911  * add a clique between both and-resultants, negated variables are not save in cliquetables
4912  *
4913  * set r_1 = var1 * z_1; (z_1 is some product)
4914  * set r_2 = var2 * z_2; (z_2 is some product)
4915  * var1 == ~var2
4916  *
4917  * if:
4918  * var1 + ~var1 <= 1; r_1 r_2
4919  * 0 + 1 <= 1 0 1 or 0 \
4920  * 1 + 0 <= 1 ==> 1 or 0 0 > ==> r_1 + r_2 <= 1
4921  * 0 + 0 <= 1 0 0 /
4922  */
4923  if( values[0] != values[1] && var1 == var2 )
4924  {
4925  SCIP_CONS* newcons;
4926  SCIP_VAR* clqvars[2];
4927  char consname[SCIP_MAXSTRLEN];
4928 
4929  clqvars[0] = andres;
4930  clqvars[1] = andres2;
4931 
4932  /* @todo: check whether it is better to only add the clique or to add the setppc constraint or do both */
4933 
4934  /* add clique */
4935  SCIP_CALL( SCIPaddClique(scip, clqvars, NULL, 2, FALSE, cutoff, &nchgbdslocal) );
4936  if( *cutoff )
4937  goto TERMINATE;
4938 
4939  *nchgbds += nchgbdslocal;
4940 
4941  (void) SCIPsnprintf(consname, SCIP_MAXSTRLEN, "%s_clq_%s_%s", SCIPconsGetName(cons), SCIPvarGetName(clqvars[0]), SCIPvarGetName(clqvars[1]) );
4942  SCIP_CALL( SCIPcreateConsSetpack(scip, &newcons, consname, 2, clqvars,
4944  FALSE, SCIPconsIsPropagated(cons),
4947 
4948  SCIP_CALL( SCIPaddCons(scip, newcons) );
4949  SCIPdebugMsg(scip, "added a clique/setppc constraint <%s> \n", SCIPconsGetName(newcons));
4950  SCIPdebugPrintCons(scip, newcons, NULL);
4951 
4952  SCIP_CALL( SCIPreleaseCons(scip, &newcons) );
4953  }
4954  /* if a variable in an and-constraint is in a clique with a variable in another and-constraint, we can add
4955  * the clique between both and-resultant
4956  *
4957  * let r_1 = var1 * z_1; (z_1 is some product)
4958  * let r_2 = var2 * z_2; (z_2 is some product)
4959  *
4960  * if:
4961  * var1 + var2 <= 1; r_1 r_2
4962  * 0 + 1 <= 1 0 1 or 0 \
4963  * 1 + 0 <= 1 ==> 1 or 0 0 > ==> r_1 + r_2 <= 1
4964  * 0 + 0 <= 1 0 0 /
4965  */
4966  else if( SCIPvarsHaveCommonClique(var1, values[0], var2, values[1], TRUE) && (var1 != var2) )
4967  {
4968  SCIP_CONS* newcons;
4969  SCIP_VAR* clqvars[2];
4970  char consname[SCIP_MAXSTRLEN];
4971 
4972  clqvars[0] = andres;
4973  clqvars[1] = values[1] ? var2 : SCIPvarGetNegatedVar(var2);
4974  assert(clqvars[1] != NULL);
4975 
4976  /* @todo: check whether it is better to only add the clique or to add the setppc constraint or do both */
4977 
4978  /* add clique */
4979  SCIP_CALL( SCIPaddClique(scip, clqvars, NULL, 2, FALSE, cutoff, &nchgbdslocal) );
4980  if( *cutoff )
4981  goto TERMINATE;
4982 
4983  *nchgbds += nchgbdslocal;
4984 
4985  (void) SCIPsnprintf(consname, SCIP_MAXSTRLEN, "%s_clq_%s_%s", SCIPconsGetName(cons), SCIPvarGetName(clqvars[0]), SCIPvarGetName(clqvars[1]) );
4986  SCIP_CALL( SCIPcreateConsSetpack(scip, &newcons, consname, 2, clqvars,
4988  FALSE, SCIPconsIsPropagated(cons),
4991 
4992  SCIP_CALL( SCIPaddCons(scip, newcons) );
4993  SCIPdebugMsg(scip, "added a clique/setppc constraint <%s> \n", SCIPconsGetName(newcons));
4994  SCIPdebugPrintCons(scip, newcons, NULL);
4995 
4996  SCIP_CALL( SCIPreleaseCons(scip, &newcons) );
4997  }
4998  }
4999  }
5000  }
5001 
5002  TERMINATE:
5003  /* free temporary memory */
5004  SCIPfreeBufferArray(scip, &linvars);
5005  SCIPfreeBufferArray(scip, &vars);
5006 
5007  return SCIP_OKAY;
5008 }
5009 
5010 /** propagation method for pseudoboolean constraints */
5011 static
5013  SCIP*const scip, /**< SCIP data structure */
5014  SCIP_CONS*const cons, /**< knapsack constraint */
5015  SCIP_Bool*const cutoff, /**< pointer to store whether the node can be cut off */
5016  int*const ndelconss /**< pointer to count number of deleted constraints */
5017  )
5018 {
5019  SCIP_CONSDATA* consdata;
5020 
5021  assert(scip != NULL);
5022  assert(cons != NULL);
5023  assert(cutoff != NULL);
5024  assert(ndelconss != NULL);
5025 
5026  *cutoff = FALSE;
5027 
5028  consdata = SCIPconsGetData(cons);
5029  assert(consdata != NULL);
5030  assert(consdata->lincons != NULL);
5031 
5032  /* if linear constraint is redundant, than pseudoboolean constraint is redundant too */
5033  if( SCIPconsIsDeleted(consdata->lincons) )
5034  {
5035  SCIP_CALL( SCIPdelConsLocal(scip, cons) );
5036  ++(*ndelconss);
5037  }
5038 
5039  /* check if the constraint was already propagated */
5040  if( consdata->propagated )
5041  return SCIP_OKAY;
5042 
5043  /* mark the constraint propagated */
5044  consdata->propagated = TRUE;
5045 
5046  return SCIP_OKAY;
5047 }
5048 
5049 /** update and-constraint flags due to pseudoboolean constraint flags */
5050 static
5052  SCIP*const scip, /**< SCIP data structure */
5053  SCIP_CONS*const cons /**< pseudoboolean constraint */
5054  )
5055 {
5056  CONSANDDATA** consanddatas;
5057  int nconsanddatas;
5058  SCIP_CONSDATA* consdata;
5059  int c;
5060 
5061  assert(scip != NULL);
5062  assert(cons != NULL);
5063 
5064  consdata = SCIPconsGetData(cons);
5065  assert(consdata != NULL);
5066 
5067  consanddatas = consdata->consanddatas;
5068  nconsanddatas = consdata->nconsanddatas;
5069  assert(nconsanddatas == 0 || consanddatas != NULL);
5070 
5071  if( !SCIPconsIsActive(cons) )
5072  return SCIP_OKAY;
5073 
5074  /* release and-constraints and change check flag of and-constraint */
5075  for( c = nconsanddatas - 1; c >= 0; --c )
5076  {
5077  SCIP_CONS* andcons;
5078 
5079  assert(consanddatas[c] != NULL);
5080 
5081  if( !consanddatas[c]->istransformed )
5082  continue;
5083 
5084  andcons = consanddatas[c]->cons;
5085  assert(andcons != NULL);
5086 
5087  SCIP_CALL( SCIPsetConsChecked(scip, andcons, SCIPconsIsChecked(cons)) );
5088  }
5089 
5090  return SCIP_OKAY;
5091 }
5092 
5093 /** delete unused information in constraint handler data */
5094 static
5096  SCIP*const scip, /**< SCIP data structure */
5097  SCIP_CONSHDLRDATA*const conshdlrdata, /**< pseudoboolean constraint handler data */
5098  int*const ndelconss /**< pointer to count number of deleted constraints */
5099  )
5100 {
5101  CONSANDDATA** allconsanddatas;
5102  CONSANDDATA* consanddata;
5103  int c;
5104 
5105  assert(scip != NULL);
5106  assert(conshdlrdata != NULL);
5107  assert(ndelconss != NULL);
5108 
5109  allconsanddatas = conshdlrdata->allconsanddatas;
5110  assert(allconsanddatas != NULL);
5111  assert(conshdlrdata->nallconsanddatas > 0);
5112  assert(conshdlrdata->nallconsanddatas <= conshdlrdata->sallconsanddatas);
5113 
5114  for( c = conshdlrdata->nallconsanddatas - 1; c >= 0; --c )
5115  {
5116  SCIP_VAR** tmpvars;
5117  int stmpvars;
5118  SCIP_CONS* cons;
5119  int v;
5120 
5121  consanddata = allconsanddatas[c];
5122 
5123  assert(consanddata->nvars == 0 || (consanddata->vars != NULL && consanddata->svars > 0));
5124  assert(consanddata->nnewvars == 0 || (consanddata->newvars != NULL && consanddata->snewvars > 0));
5125 
5126  if( !consanddata->istransformed )
5127  {
5128  assert(consanddata->vars == NULL || consanddata->origcons != NULL);
5129  assert(consanddata->nvars == 0 || consanddata->origcons != NULL);
5130  assert(consanddata->svars == 0 || consanddata->origcons != NULL);
5131  assert(consanddata->newvars == NULL);
5132  assert(consanddata->nnewvars == 0);
5133  assert(consanddata->snewvars == 0);
5134 
5135  continue;
5136  }
5137 
5138  /* if no variables are left, delete variables arrays */
5139  if( consanddata->nvars == 0 )
5140  {
5141  SCIP_VAR* resvar = SCIPgetResultantAnd(scip, consanddata->cons);
5142 
5143  /* if we have no old variables, than also no new variables */
5144  assert(consanddata->nnewvars == 0);
5145  assert(consanddata->nuses > 0);
5146  assert(resvar != NULL);
5147 
5148  /* delete and-constraint */
5149  SCIP_CALL( SCIPdelCons(scip, consanddata->cons) );
5150  ++(*ndelconss);
5151 
5152  SCIP_CALL( transformToOrig(scip, consanddata, conshdlrdata) );
5153 
5154  /* release and-constraint */
5155  SCIP_CALL( SCIPreleaseCons(scip, &consanddata->cons) );
5156  consanddata->nuses = 0;
5157 
5158  /* remove consanddata from hashtable, if it existed only in transformed space */
5159  if( consanddata->origcons == NULL )
5160  {
5161  assert(SCIPhashtableExists(conshdlrdata->hashtable, (void*)consanddata));
5162  SCIP_CALL( SCIPhashtableRemove(conshdlrdata->hashtable, (void*)consanddata) );
5163  }
5164  assert(SCIPhashmapExists(conshdlrdata->hashmap, (void*)resvar));
5165  SCIP_CALL( SCIPhashmapRemove(conshdlrdata->hashmap, (void*)resvar) );
5166 
5167  continue;
5168  }
5169 
5170  /* the consanddata object is not used anymore, so extract the and constraint and delete other data */
5171  if( consanddata->nuses == 0 )
5172  {
5173  SCIP_Bool looseorcolumn;
5174  SCIP_VARSTATUS varstatus;
5175 
5176  if( consanddata->cons == NULL )
5177  {
5178  assert(!consanddata->istransformed || consanddata->noriguses > 0);
5179  assert((consanddata->noriguses > 0) == (consanddata->origcons != NULL));
5180  assert(consanddata->vars == NULL || consanddata->origcons != NULL);
5181  assert(consanddata->nvars == 0 || consanddata->origcons != NULL);
5182  assert(consanddata->svars == 0 || consanddata->origcons != NULL);
5183  assert(consanddata->newvars == NULL);
5184  assert(consanddata->nnewvars == 0);
5185  assert(consanddata->snewvars == 0);
5186 
5187  continue;
5188  }
5189 
5190  SCIP_CALL( transformToOrig(scip, consanddata, conshdlrdata) );
5191 
5192  varstatus = SCIPvarGetStatus(SCIPgetResultantAnd(scip, consanddata->cons));
5193  looseorcolumn = (varstatus == SCIP_VARSTATUS_LOOSE || varstatus == SCIP_VARSTATUS_COLUMN);
5194 
5195 #if 1
5196  /* @note due to aggregations or fixings the resultant may need to be propagated later on, so we can only
5197  * delete the and-constraint if the resultant is of column or loose status
5198  * and is not an active variable of another (multi-)aggregated/negated variable
5199  */
5200  if( looseorcolumn )
5201  {
5202  SCIP_Bool del = TRUE;
5203  int nfixedvars = SCIPgetNFixedVars(scip);
5204 
5205  if( nfixedvars > 0 )
5206  {
5207  SCIP_VAR** fixedvars;
5208  SCIP_VAR** scipfixedvars;
5209  SCIP_VAR** activevars = NULL;
5210  SCIP_Real* activescalars = NULL;
5211  SCIP_Real activeconstant;
5212  int nactivevars;
5213  int requiredsize;
5214  int pos;
5215  int w;
5216 
5217  scipfixedvars = SCIPgetFixedVars(scip);
5218  SCIP_CALL( SCIPduplicateBufferArray(scip, &fixedvars, scipfixedvars, nfixedvars) );
5219 
5220  SCIPvarsGetProbvar(fixedvars, nfixedvars);
5221 
5222  /* all inactive variables have a loose, column, fixed or multi-aggregated variable as counterpart,
5223  * for multi-aggregated variables, we need to check all active representatives
5224  * @todo move this outside of the consanddata loop
5225  */
5226  for( w = nfixedvars - 1; w >= 0; --w )
5227  {
5228  if( SCIPvarGetStatus(fixedvars[w]) == SCIP_VARSTATUS_MULTAGGR )
5229  {
5230  if( activevars == NULL )
5231  {
5232  SCIP_CALL( SCIPallocBufferArray(scip, &activevars, SCIPgetNVars(scip)) );
5233  SCIP_CALL( SCIPallocBufferArray(scip, &activescalars, SCIPgetNVars(scip)) );
5234  }
5235  assert(activevars != NULL);
5236  assert(activescalars != NULL);
5237 
5238  activevars[0] = fixedvars[w];
5239  activescalars[0] = 1.0;
5240  activeconstant = 0.0;
5241  nactivevars = 1;
5242 
5243  SCIP_CALL( SCIPgetProbvarLinearSum(scip, activevars, activescalars, &nactivevars, SCIPgetNVars(scip),
5244  &activeconstant, &requiredsize, TRUE) );
5245  assert(requiredsize <= SCIPgetNVars(scip));
5246 
5247  if( nactivevars == 0 )
5248  {
5249  --nfixedvars;
5250  fixedvars[w] = fixedvars[nfixedvars];
5251  }
5252  else
5253  {
5254  fixedvars[w] = activevars[0];
5255 
5256  if( nactivevars > 1 )
5257  {
5258  int i;
5259 
5260  SCIP_CALL( SCIPreallocBufferArray(scip, &fixedvars, nfixedvars + nactivevars - 1) );
5261  for( i = 1; i < nactivevars; ++i )
5262  {
5263  assert(SCIPvarGetStatus(activevars[i]) == SCIP_VARSTATUS_LOOSE || SCIPvarGetStatus(activevars[i]) == SCIP_VARSTATUS_COLUMN || SCIPvarGetStatus(activevars[i]) == SCIP_VARSTATUS_FIXED);
5264  fixedvars[nfixedvars] = activevars[i];
5265  ++nfixedvars;
5266  }
5267  }
5268  }
5269  }
5270 
5271  assert(SCIPvarGetStatus(fixedvars[w]) == SCIP_VARSTATUS_LOOSE || SCIPvarGetStatus(fixedvars[w]) == SCIP_VARSTATUS_COLUMN || SCIPvarGetStatus(fixedvars[w]) == SCIP_VARSTATUS_FIXED);
5272  }
5273 
5274  if( activevars != NULL )
5275  {
5276  SCIPfreeBufferArray(scip, &activevars);
5277  SCIPfreeBufferArray(scip, &activescalars);
5278  }
5279 
5280  SCIPsortPtr((void**)fixedvars, SCIPvarComp, nfixedvars);
5281 
5282  if( SCIPsortedvecFindPtr((void**)fixedvars, SCIPvarComp, SCIPgetResultantAnd(scip, consanddata->cons), nfixedvars, &pos) )
5283  del = FALSE;
5284 
5285  SCIPfreeBufferArray(scip, &fixedvars);
5286  }
5287 
5288  if( del )
5289  {
5290  SCIP_CALL( SCIPdelCons(scip, consanddata->cons) );
5291  }
5292  }
5293 #endif
5294 
5295  if( !SCIPconsIsDeleted(consanddata->cons) )
5296  {
5297  /* change flags */
5298  if( !looseorcolumn )
5299  {
5300  SCIP_CALL( SCIPsetConsInitial(scip, consanddata->cons, FALSE) );
5301 #if 0
5302  SCIP_CALL( SCIPsetConsSeparated(scip, consanddata->cons, FALSE) );
5303 #endif
5304  }
5305  SCIP_CALL( SCIPsetConsChecked(scip, consanddata->cons, TRUE) );
5306  }
5307 
5308  /* remove consanddata from hashtable, if it existed only in transformed space */
5309  if( consanddata->origcons == NULL )
5310  {
5311  assert(SCIPhashtableExists(conshdlrdata->hashtable, (void*)consanddata));
5312  SCIP_CALL( SCIPhashtableRemove(conshdlrdata->hashtable, (void*)consanddata) );
5313  }
5314  assert(SCIPhashmapExists(conshdlrdata->hashmap, (void*)SCIPgetResultantAnd(scip, consanddata->cons)));
5315  SCIP_CALL( SCIPhashmapRemove(conshdlrdata->hashmap, (void*)SCIPgetResultantAnd(scip, consanddata->cons)) );
5316 
5317  SCIP_CALL( SCIPreleaseCons(scip, &(consanddata->cons)) );
5318  ++(*ndelconss);
5319 
5320  continue;
5321  }
5322 
5323  cons = consanddata->cons;
5324  assert(cons != NULL);
5325 
5326  /* if and-constraint is deleted, delete variables arrays */
5327  if( SCIPconsIsDeleted(cons) )
5328  {
5329  SCIP_VAR* resvar = SCIPgetResultantAnd(scip, consanddata->cons);
5330 
5331  assert(consanddata->nuses > 0);
5332  assert(resvar != NULL);
5333 
5334  SCIP_CALL( transformToOrig(scip, consanddata, conshdlrdata) );
5335 
5336  /* release and-constraint */
5337  SCIP_CALL( SCIPreleaseCons(scip, &consanddata->cons) );
5338  consanddata->nuses = 0;
5339 
5340  /* remove consanddata from hashtable, if it existed only in transformed space */
5341  if( consanddata->origcons == NULL )
5342  {
5343  assert(SCIPhashtableExists(conshdlrdata->hashtable, (void*)consanddata));
5344  SCIP_CALL( SCIPhashtableRemove(conshdlrdata->hashtable, (void*)consanddata) );
5345  }
5346  assert(SCIPhashmapExists(conshdlrdata->hashmap, (void*)resvar));
5347  SCIP_CALL( SCIPhashmapRemove(conshdlrdata->hashmap, (void*)resvar) );
5348 
5349  continue;
5350  }
5351 
5352  /* if no new variables exist, we do not need to do anything here */
5353  if( consanddata->nnewvars == 0 )
5354  continue;
5355 
5356  tmpvars = consanddata->vars;
5357  /* release all variables */
5358  for( v = consanddata->nvars - 1; v >= 0; --v )
5359  {
5360  /* in original problem the variables was already deleted */
5361  assert(tmpvars[v] != NULL);
5362  SCIP_CALL( SCIPreleaseVar(scip, &tmpvars[v]) );
5363  }
5364 
5365  /* exchange newvars with old vars array */
5366  tmpvars = consanddata->vars;
5367  stmpvars = consanddata->svars;
5368  consanddata->vars = consanddata->newvars;
5369  consanddata->svars = consanddata->snewvars;
5370  consanddata->nvars = consanddata->nnewvars;
5371  consanddata->newvars = tmpvars;
5372  consanddata->snewvars = stmpvars;
5373  /* reset number of variables in newvars array */
5374  consanddata->nnewvars = 0;
5375  }
5376 
5377  return SCIP_OKAY;
5378 }
5379 
5380 /** update the uses counter of consandata objects which are used in pseudoboolean constraint, that were deleted and
5381  * probably delete and-constraints
5382  */
5383 static
5385  SCIP*const scip, /**< SCIP data structure */
5386  SCIP_CONS*const cons, /**< pseudoboolean constraint */
5387  SCIP_CONSHDLRDATA*const conshdlrdata, /**< pseudoboolean constraint handler data */
5388  int*const ndelconss /**< pointer to store number of deleted constraints */
5389  )
5390 {
5391  CONSANDDATA** consanddatas;
5392  int nconsanddatas;
5393  SCIP_CONSDATA* consdata;
5394  int c;
5395 
5396  assert(scip != NULL);
5397  assert(cons != NULL);
5398  assert(conshdlrdata != NULL);
5399  assert(ndelconss != NULL);
5400 
5401  /* can only be called when constraint was deleted */
5402  assert(SCIPconsIsDeleted(cons));
5403 
5404  consdata = SCIPconsGetData(cons);
5405  assert(consdata != NULL);
5406 
5407  consanddatas = consdata->consanddatas;
5408  nconsanddatas = consdata->nconsanddatas;
5409  assert(nconsanddatas > 0 && consanddatas != NULL);
5410  assert(consdata->andcoefs != NULL);
5411 
5412  /* remove old locks */
5413  for( c = nconsanddatas - 1; c >= 0; --c )
5414  {
5415  CONSANDDATA* consanddata;
5416 
5417  consanddata = consanddatas[c];
5418  assert(consanddata != NULL);
5419 
5420  if( !consanddata->istransformed )
5421  continue;
5422 
5423  SCIP_CALL( removeOldLocks(scip, cons, consanddata, consdata->andcoefs[c], consdata->lhs, consdata->rhs) );
5424  }
5425 
5426  /* correct consandata usage counters and data */
5427  for( c = nconsanddatas - 1; c >= 0; --c )
5428  {
5429  CONSANDDATA* consanddata;
5430 
5431  consanddata = consanddatas[c];
5432  assert(consanddata != NULL);
5433  assert(consanddatas[c]->istransformed);
5434 
5435  assert(consanddata->nuses > 0);
5436 
5437  if( consanddata->nuses > 0 )
5438  --(consanddata->nuses);
5439 
5440  /* if data object is not used anymore, delete it */
5441  if( consanddata->nuses == 0 )
5442  {
5443  SCIP_VAR* resvar;
5444  SCIP_VARSTATUS varstatus;
5445  SCIP_Bool looseorcolumn;
5446 
5447  SCIP_CALL( transformToOrig(scip, consanddata, conshdlrdata) );
5448 
5449  resvar = SCIPgetResultantAnd(scip, consanddata->cons);
5450  assert(resvar != NULL);
5451 
5452  varstatus = SCIPvarGetStatus(resvar);
5453  looseorcolumn = (varstatus == SCIP_VARSTATUS_LOOSE || varstatus == SCIP_VARSTATUS_COLUMN);
5454 
5455 #if 1
5456  /* @note due to aggregations or fixings the resultant may need to be propagated later on, so we can only
5457  * delete the and-constraint if the resultant is of column or loose status
5458  * and is not an active variable of another (multi-)aggregated/negated variable
5459  */
5460  if( looseorcolumn )
5461  {
5462  SCIP_Bool delcons = TRUE;
5463 #if 0
5464  const int nfixedvars = SCIPgetNFixedVars(scip);
5465 
5466  if( nfixedvars > 0 )
5467  {
5468  SCIP_VAR** fixedvars;
5469  SCIP_Bool foundmultiaggrvar = FALSE; /* workaround for multi-aggregated variables */
5470  int pos;
5471  int w;
5472 
5473  SCIP_CALL( SCIPduplicateBufferArray(scip, &fixedvars, SCIPgetFixedVars(scip), nfixedvars) );
5474 
5475  SCIPvarsGetProbvar(fixedvars, nfixedvars);
5476 
5477  /* all inactive variables have a loose, column, fixed or multi-aggregated variable as counterpart, but
5478  * because we have only binary variables (in pseudobbolean contest) there should also be no
5479  * multi-aggregated variable
5480  *
5481  * @todo for multi-aggregated variables check also all active representatives for this resultant
5482  */
5483  for( w = nfixedvars - 1; w >= 0; --w )
5484  {
5485  if( SCIPvarGetStatus(fixedvars[w]) == SCIP_VARSTATUS_MULTAGGR )
5486  foundmultiaggrvar = TRUE;
5487  else
5488  assert(SCIPvarGetStatus(fixedvars[w]) == SCIP_VARSTATUS_LOOSE || SCIPvarGetStatus(fixedvars[w]) == SCIP_VARSTATUS_COLUMN || SCIPvarGetStatus(fixedvars[w]) == SCIP_VARSTATUS_FIXED);
5489  }
5490 
5491  SCIPsortPtr((void**)fixedvars, SCIPvarComp, nfixedvars);
5492 
5493  if( foundmultiaggrvar )
5494  delcons = FALSE;
5495  else if( SCIPsortedvecFindPtr((void**)fixedvars, SCIPvarComp, resvar, nfixedvars, &pos) )
5496  delcons = FALSE;
5497 
5498  SCIPfreeBufferArray(scip, &fixedvars);
5499  }
5500 #endif
5501  /* we can only delete and constraints if the resultant is an artificial variable and also active, because
5502  * then the assigned value is not of interest and the artificial and constraint does not need to be
5503  * fulfilled
5504  *
5505  * if this variable is not such an artificial variable we need the IRRELEVANT vartype which should be the
5506  * correct way to fix this
5507  */
5508  if( delcons
5509 #if 0
5510  && strlen(SCIPvarGetName(resvar)) > strlen(ARTIFICIALVARNAMEPREFIX) &&
5511  strncmp(SCIPvarGetName(resvar)+2, ARTIFICIALVARNAMEPREFIX, strlen(ARTIFICIALVARNAMEPREFIX)) == 0
5512 #endif
5513  ) /*lint !e774*/
5514  {
5515  assert(!SCIPconsIsChecked(consanddata->cons));
5516  SCIP_CALL( SCIPdelCons(scip, consanddata->cons) );
5517  }
5518  }
5519 #endif
5520 
5521 #if 0
5522  /* @note due to aggregations or fixings the resultant may need to be propagated later on, so we can only
5523  * delete the and-constraint if the resultant is of column or loose status
5524  * and is not an active variable of another (multi-)aggregated/negated variable
5525  */
5526  if( looseorcolumn )
5527  {
5528  SCIP_CALL( SCIPdelCons(scip, consanddata->cons) );
5529  }
5530 #endif
5531 
5532  if( !SCIPconsIsDeleted(consanddata->cons) )
5533  {
5534  /* change flags */
5535  if( !looseorcolumn )
5536  {
5537  SCIP_CALL( SCIPsetConsInitial(scip, consanddata->cons, FALSE) );
5538 #if 0
5539  SCIP_CALL( SCIPsetConsSeparated(scip, consanddata->cons, FALSE) );
5540 #endif
5541  }
5542  SCIP_CALL( SCIPsetConsChecked(scip, consanddata->cons, TRUE) );
5543  }
5544 
5545  /* remove consanddata from hashtable, if it existed only in transformed space */
5546  if( consanddata->origcons == NULL )
5547  {
5548  assert(SCIPhashtableExists(conshdlrdata->hashtable, (void*)consanddata));
5549  SCIP_CALL( SCIPhashtableRemove(conshdlrdata->hashtable, (void*)consanddata) );
5550  }
5551  assert(SCIPhashmapExists(conshdlrdata->hashmap, (void*)SCIPgetResultantAnd(scip, consanddata->cons)));
5552  SCIP_CALL( SCIPhashmapRemove(conshdlrdata->hashmap, (void*)SCIPgetResultantAnd(scip, consanddata->cons)) );
5553 
5554  SCIP_CALL( SCIPreleaseCons(scip, &(consanddata->cons)) );
5555  ++(*ndelconss);
5556  }
5557  }
5558 
5559  consdata->nconsanddatas = 0;
5560 
5561  return SCIP_OKAY;
5562 }
5563 
5564 
5565 /* maximal number to enumerate solutions for one pseudoboolean constraint to check for an upgrade to an XOR constraint */
5566 #define MAXNVARS 10 /* note that this cannot be bigger than 31 */
5568 /** calculate result for a given pseudoboolean constraint with given values, this is used to decide whether a
5569  * pseudoboolean constraint can be upgrade to an XOR constraint
5570  */
5571 static
5573  SCIP*const scip, /**< SCIP data structure */
5574  SCIP_VAR**const vars, /**< all variables which occur */
5575  int const nvars, /**< number of all variables which appear in the pseudoboolean
5576  * constraint
5577  */
5578  SCIP_Bool*const values, /**< values of all variables which appear in the pseudoboolean
5579  * constraint
5580  */
5581  SCIP_VAR**const linvars, /**< linear variables */
5582  SCIP_Real*const lincoefs, /**< linear coefficients */
5583  int const nlinvars, /**< number of linear variables */
5584  SCIP_Real const constant, /**< offset to the linear part */
5585  SCIP_Real const side, /**< side of pseudoboolean constraint */
5586  CONSANDDATA**const consanddatas, /**< all consanddata objects in a constraint */
5587  SCIP_Real*const consanddatacoefs, /**< nonlinear coefficients */
5588  SCIP_Bool*const consanddatanegs, /**< negation status of and resultants in pseudo-boolean constraint */
5589  int const nconsanddatas, /**< number of all consanddata objects */
5590  int const cnt, /**< number of variables set to 1 */
5591  int*const xortype /**< pointer to save the possible xor type if a solution was valid and does
5592  * not violate the old xortype
5593  */
5594  )
5595 {
5596  CONSANDDATA* consanddata;
5597  SCIP_VAR** termvars;
5598  SCIP_VAR** repvars;
5599  int ntermvars;
5600  SCIP_Bool* negated;
5601  SCIP_Real value;
5602  int pos;
5603  int v;
5604  int c;
5605 
5606  assert(scip != NULL);
5607  assert(vars != NULL);
5608  assert(nvars > 0);
5609  assert(values != NULL);
5610  assert(linvars != NULL || nlinvars == 0);
5611  assert(lincoefs != NULL || nlinvars == 0);
5612  assert(nvars >= nlinvars);
5613  assert(SCIPisEQ(scip, side, 1.0) || SCIPisZero(scip, side));
5614  assert(consanddatas != NULL);
5615  assert(consanddatacoefs != NULL);
5616  assert(nconsanddatas > 0);
5617  assert(*xortype >= -1 && *xortype <= 1);
5618 
5619  /* order the variables after index, to compare them easier */
5620  SCIPsortPtr((void**)linvars, SCIPvarCompActiveAndNegated, nlinvars);
5621  SCIPsortPtr((void**)vars, SCIPvarCompActiveAndNegated, nvars);
5622 
5623  value = constant;
5624  for( v = nlinvars - 1; v >= 0; --v )
5625  {
5626  if( SCIPsortedvecFindPtr((void**)vars, SCIPvarCompActiveAndNegated, linvars[v], nvars, &pos) ) /*lint !e613*/
5627  {
5628  if( values[pos] )
5629  value += lincoefs[v]; /*lint !e613*/
5630  }
5631  else
5632  {
5633  /* this cannot happen, all linear variables should be a part of 'vars' */
5634  SCIPABORT();
5635 
5636  *xortype = -1; /*lint !e527*/
5637  return SCIP_OKAY;
5638  }
5639  }
5640 
5641  SCIP_CALL( SCIPallocBufferArray(scip, &repvars, MAXNVARS) );
5642  SCIP_CALL( SCIPallocBufferArray(scip, &negated, MAXNVARS) );
5643 
5644  for( c = nconsanddatas - 1; c >= 0; --c )
5645  {
5646  SCIP_Bool val = TRUE;
5647 
5648  consanddata = consanddatas[c];
5649  assert(consanddata != NULL);
5650  assert(consanddata->istransformed);
5651 
5652  /* choose correct variable array to add locks for, we only add locks for now valid variables */
5653  if( consanddata->nnewvars > 0 )
5654  {
5655  termvars = consanddata->newvars;
5656  ntermvars = consanddata->nnewvars;
5657  }
5658  else
5659