Scippy

SCIP

Solving Constraint Integer Programs

cons_countsols.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-2024 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_countsols.c
26 * @ingroup DEFPLUGINS_CONS
27 * @brief constraint handler for counting feasible solutions
28 * @author Stefan Heinz
29 * @author Michael Winkler
30 *
31 * If this constraint handler is activated than it counts or collects all feasible solutions. We refer to \ref COUNTER for
32 * more details about using SCIP for counting feasible solutions.
33 *
34 * @todo In the last round of presolving we should check if variables exist, which have up and down lock one. In this
35 * case we know that these locks are coming from this constraint handler. Therefore, they are totally free and can
36 * be ignored in the branch and bound process. To get this result we have to store these variables in the
37 * constraint handler data structure (to remember this free dimensions) and fix them to any feasible value.
38 */
39
40/*---+----1----+----2----+----3----+----4----+----5----+----6----+----7----+----8----+----9----+----0----+----1----+----2*/
41
44#include "scip/cons_countsols.h"
45#include "scip/cons_knapsack.h"
46#include "scip/cons_logicor.h"
47#include "scip/cons_setppc.h"
48#include "scip/cons_varbound.h"
49#include "scip/dialog_default.h"
50#include "scip/pub_cons.h"
51#include "scip/pub_dialog.h"
52#include "scip/pub_disp.h"
53#include "scip/pub_heur.h"
54#include "scip/pub_message.h"
55#include "scip/pub_misc.h"
56#include "scip/pub_misc_sort.h"
57#include "scip/pub_sol.h"
58#include "scip/pub_var.h"
59#include "scip/scip_branch.h"
60#include "scip/scip_cons.h"
61#include "scip/scip_dialog.h"
62#include "scip/scip_disp.h"
63#include "scip/scip_general.h"
64#include "scip/scip_heur.h"
65#include "scip/scip_mem.h"
66#include "scip/scip_message.h"
67#include "scip/scip_numerics.h"
68#include "scip/scip_param.h"
69#include "scip/scip_prob.h"
70#include "scip/scip_sol.h"
71#include "scip/scip_solve.h"
72#include "scip/scip_var.h"
74#include <string.h>
75
76/* depending on whether the GMP library is available we use a GMP data type or a SCIP_Longint */
77#ifdef SCIP_WITH_GMP
78#include <gmp.h>
79typedef mpz_t Int;
80#else
82#endif
83
84/* constraint handler properties */
85#define CONSHDLR_NAME "countsols"
86#define CONSHDLR_DESC "constraint to count feasible solutions"
87#define CONSHDLR_ENFOPRIORITY -9999999 /**< priority of the constraint handler for constraint enforcing */
88#define CONSHDLR_CHECKPRIORITY -9999999 /**< priority of the constraint handler for checking feasibility */
89#define CONSHDLR_EAGERFREQ 100 /**< frequency for using all instead of only the useful constraints in separation,
90 * propagation and enforcement, -1 for no eager evaluations, 0 for first only */
91#define CONSHDLR_NEEDSCONS FALSE /**< should the constraint handler be skipped, if no constraints are available? */
92
93/* default parameter settings */
94#define DEFAULT_SPARSETEST TRUE /**< sparse test on or off */
95#define DEFAULT_DISCARDSOLS TRUE /**< is it allowed to discard solutions */
96#define DEFAULT_ACTIVE FALSE /**< is the constraint handler active */
97#define DEFAULT_COLLECT FALSE /**< should the solutions be collected */
98#define DEFAULT_SOLLIMIT -1LL /**< counting stops, if the given number of solutions were found (-1: no limit) */
99
100/* default column settings */
101#define DISP_SOLS_NAME "sols"
102#define DISP_SOLS_DESC "number of detected feasible solutions"
103#define DISP_SOLS_HEADER " sols "
104#define DISP_SOLS_WIDTH 7
105#define DISP_SOLS_PRIORITY 110000
106#define DISP_SOLS_POSITION 100000
107#define DISP_SOLS_STRIPLINE TRUE
108
109#define DISP_CUTS_NAME "feasST"
110#define DISP_CUTS_DESC "number of detected non trivial feasible subtrees"
111#define DISP_CUTS_HEADER "feasST"
112#define DISP_CUTS_WIDTH 6
113#define DISP_CUTS_PRIORITY 110000
114#define DISP_CUTS_POSITION 110000
115#define DISP_CUTS_STRIPLINE TRUE
116
117/** creates and adds a constraint which cuts off the solution from the feasibility region
118 *
119 * input:
120 * - scip : SCIP main data structure
121 * - sol : solution to cut off
122 * - conshdlrdata : constraint handler data
123 */
124#define CUTOFF_CONSTRAINT(x) SCIP_RETCODE x (SCIP* scip, SCIP_SOL* sol, SCIP_CONSHDLRDATA* conshdlrdata)
125
126
127/** constraint handler data */
128struct SCIP_ConshdlrData
129{
130 /* solution data and statistic variables */
131 SCIP_SPARSESOL** solutions; /**< array to store all solutions */
132 int nsolutions; /**< number of solution stored */
133 int ssolutions; /**< size of the solution array */
134 int feasST; /**< number of non trivial feasible subtrees */
135 int nDiscardSols; /**< number of discarded solutions */
136 int nNonSparseSols; /**< number of non sparse solutions */
137 Int nsols; /**< number of solutions */
138 CUTOFF_CONSTRAINT((*cutoffSolution)); /**< method for cutting of a solution */
139
140 /* constraint handler parameters */
141 SCIP_Longint sollimit; /**< counting stops, if the given number of solutions have been found (-1: no limit) */
142 SCIP_Bool active; /**< constraint handler active */
143 SCIP_Bool discardsols; /**< allow to discard solutions */
144 SCIP_Bool sparsetest; /**< allow to check for sparse solutions */
145 SCIP_Bool collect; /**< should the solutions be collected */
146
147 SCIP_Bool warning; /**< has the warning message already been posted? */
148
149 /* specific problem data */
150 SCIP_HASHMAP* hashmap; /**< hashmap to store position of active transformed problem variable in our vars array */
151 SCIP_VAR** allvars; /**< array containing a copy of all variables before presolving */
152 SCIP_VAR** vars; /**< array containing a copy of all active variables (after presolving) */
153 int nallvars; /**< number of all variables in the problem */
154 int nvars; /**< number of all active variables in the problem */
155 SCIP_Bool continuous; /**< are there continuous variables */
156};
157
158
159/*
160 * Local methods for handling the <Int> data structure
161 */
162
163/** allocates memory for the value pointer */
164static
166 Int* value /**< pointer to the value to allocate memory */
167 )
168{ /*lint --e{715}*/
169#ifdef SCIP_WITH_GMP
170 mpz_init(*value);
171#endif
172}
173
174
175/** sets the value pointer to the new value */
176static
178 Int* value, /**< pointer to the value to initialize */
179 SCIP_Longint newvalue /**< new value */
180 )
181{
182 assert(newvalue < LONG_MAX);
183
184#ifdef SCIP_WITH_GMP
185 mpz_set_si(*value, (long) newvalue);
186#else
187 (*value) = newvalue;
188#endif
189}
190
191
192/** sets a power of 2 to the given value */
193static
195 Int* value, /**< pointer to the value to increase */
196 SCIP_Longint exponent /**< exponent for the base 2 */
197 )
198{
199 assert(0 <= exponent && exponent < LONG_MAX);
200
201#ifdef SCIP_WITH_GMP
202 mpz_ui_pow_ui(*value, 2UL, (unsigned long) exponent);
203#else
204 assert(exponent < 64);
205 (*value) = (SCIP_Longint)1 << exponent;
206#endif
207}
208
209
210/** free memory */
211static
213 Int* value /**< pointer to the value to free */
214 )
215{ /*lint --e{715}*/
216#ifdef SCIP_WITH_GMP
217 mpz_clear(*value);
218#endif
219}
220
221
222/** adds one to the given value */
223static
225 Int* value /**< pointer to the value to increase */
226 )
227{
228#ifdef SCIP_WITH_GMP
229 mpz_add_ui(*value, *value, 1UL);
230#else
231 (*value)++;
232#endif
233}
234
235
236/** adds the summand to the given value */
237static
239 Int* value, /**< pointer to the value to increase */
240 Int* summand /**< summand to add on */
241 )
242{
243#ifdef SCIP_WITH_GMP
244 mpz_add(*value, *value, *summand);
245#else
246 (*value) += (*summand);
247#endif
248}
249
250
251/** multiplies the factor by the given value */
252static
254 Int* value, /**< pointer to the value to increase */
255 SCIP_Longint factor /**< factor to multiply with */
256 )
257{
258 assert(0 <= factor && factor < LONG_MAX);
259
260#ifdef SCIP_WITH_GMP
261 mpz_mul_ui(*value, *value, (unsigned long) factor);
262#else
263 (*value) *= factor;
264#endif
265}
266
267
268/** method for creating a string out of an Int which is a mpz_t or SCIP_Longint */ /*lint -e{715}*/
269static
271 Int value, /**< number */
272 char** buffer, /**< pointer to buffer for storing the string */
273 int buffersize /**< length of the buffer */
274 )
275{ /*lint --e{715}*/
276#ifdef SCIP_WITH_GMP
277 (void) mpz_get_str(*buffer, 10, value);
278#else
279 (void) SCIPsnprintf (*buffer, buffersize, "%" SCIP_LONGINT_FORMAT "", value);
280#endif
281}
282
283
284/** method for creating a SCIP_Longing out of an Int */
285static
287 Int value, /**< number to convert */
288 SCIP_Bool* valid /**< pointer to store if the return value is valid */
289 )
290{
291#ifdef SCIP_WITH_GMP
292 *valid = FALSE;
293 if( 0 != mpz_fits_sint_p(value) )
294 (*valid) = TRUE;
295
296 return mpz_get_si(value);
297#else
298 *valid = TRUE;
299 return value;
300#endif
301}
302
303
304/*
305 * Local methods
306 */
307
308
309/** returns whether a given integer variable is unfixed in the local domain */
310static
312 SCIP_VAR* var /**< integer variable */
313 )
314{
315 assert( var != NULL );
316 assert( SCIPvarGetType(var) != SCIP_VARTYPE_CONTINUOUS );
317 assert( SCIPvarGetUbLocal(var) - SCIPvarGetLbLocal(var) >= 0.0 );
318
319 return ( SCIPvarGetUbLocal(var) - SCIPvarGetLbLocal(var) > 0.5 );
320}
321
322
323/** creates the constraint handler data */
324static
326 SCIP* scip, /**< SCIP data structure */
327 SCIP_CONSHDLRDATA** conshdlrdata /**< pointer to store constraint handler data */
328 )
329{
330 SCIP_CALL( SCIPallocBlockMemory(scip, conshdlrdata) );
331
332 (*conshdlrdata)->feasST = 0;
333 (*conshdlrdata)->nDiscardSols = 0;
334 (*conshdlrdata)->nNonSparseSols = 0;
335 (*conshdlrdata)->solutions = NULL;
336 (*conshdlrdata)->nsolutions = 0;
337 (*conshdlrdata)->ssolutions = 0;
338
339 allocInt(&(*conshdlrdata)->nsols); /*lint !e545*/
340
341 (*conshdlrdata)->cutoffSolution = NULL;
342 (*conshdlrdata)->warning = FALSE;
343 (*conshdlrdata)->hashmap = NULL;
344 (*conshdlrdata)->allvars = NULL;
345 (*conshdlrdata)->vars = NULL;
346 (*conshdlrdata)->nallvars = 0;
347 (*conshdlrdata)->nvars = 0;
348 (*conshdlrdata)->continuous = FALSE;
349
350 return SCIP_OKAY;
351}
352
353
354#ifndef NDEBUG
355/** check solution in original space */
356static
358 SCIP* scip, /**< SCIP data structure */
359 SCIP_SOL* sol, /**< solution to add */
360 SCIP_CONSHDLRDATA* conshdlrdata /**< constraint handler data */
361 )
362{
363 SCIP_Bool feasible;
364 SCIP_RETCODE retcode;
365
366 /* turn off solution counting to be able to check the solution */
367 conshdlrdata->active = FALSE;
368
369 SCIPdebugMsg(scip, "check solution in original space before counting\n");
370
371 feasible = FALSE;
372
373 /* check solution in original space */
374 retcode = SCIPcheckSolOrig(scip, sol, &feasible, TRUE, TRUE);
375 assert(feasible);
376
377 /* check return code manually */
378 if( retcode != SCIP_OKAY )
379 {
380 SCIPprintError(retcode);
381 SCIPABORT();
382 }
383
384 /* turn on solution counting to continue */
385 conshdlrdata->active = TRUE;
386}
387#else
388#define checkSolutionOrig(scip, sol, conshdlrdata) /**/
389#endif
390
391/** check if the current parameter setting is correct for a safe counting process */
392static
394 SCIP* scip /**< SCIP data structure */
395 )
396{
397 SCIP_HEUR** heuristics;
398 int nheuristics;
399 int h;
400 int intvalue;
401 SCIP_Bool valid;
402
403 assert( scip != NULL );
404
405 valid = TRUE;
406
407 /* check if all heuristics are turned off */
408 heuristics = SCIPgetHeurs(scip);
409 nheuristics = SCIPgetNHeurs(scip);
410
411 for( h = 0; h < nheuristics && valid; ++h )
412 {
413 if( SCIPheurGetFreq(heuristics[h]) != -1 )
414 valid = FALSE;
415 }
416
417 if( valid )
418 {
420 "At least one heuristic is not turned off! Heuristic solutions are currently not accepted while couting.\n");
421 }
422
423 /* check if restart is turned off */
424 SCIP_CALL( SCIPgetIntParam(scip, "presolving/maxrestarts", &intvalue) );
425 if( intvalue != 0 )
426 {
427 /* need to disable restarts, since collecting solutions won't work, but also the capturing for variables is not
428 * correctly handled
429 */
430 SCIPwarningMessage(scip, "counting forces parameter <presolving/maxrestarts> to 0.\n");
431 if( SCIPisParamFixed(scip, "presolving/maxrestarts") )
432 {
433 SCIP_CALL( SCIPunfixParam(scip, "presolving/maxrestarts") );
434 }
435
436 SCIP_CALL( SCIPsetIntParam(scip, "presolving/maxrestarts", 0) );
437 }
438
439 /* check if symmetry handling is turned off */
440 SCIP_CALL( SCIPgetIntParam(scip, "misc/usesymmetry", &intvalue) );
441 if ( intvalue != 0 )
442 {
443 /* need to disable symmetry handling, since counting is not supported if symmetry handling is enabled */
444 SCIPwarningMessage(scip, "counting forces parameter <misc/usesymmetry> to 0.\n");
445 if( SCIPisParamFixed(scip, "misc/usesymmetry") )
446 {
447 SCIP_CALL( SCIPunfixParam(scip, "misc/usesymmetry") );
448 }
449
450 SCIP_CALL( SCIPsetIntParam(scip, "misc/usesymmetry", 0) );
451 }
452
453 return SCIP_OKAY;
454}
455
456/** creates and adds a constraints which cuts off the current solution from the feasibility region in the case there are
457 * only binary variables
458 */
459static
460CUTOFF_CONSTRAINT(addBinaryCons)
461{
462 int v;
463 SCIP_VAR** consvars;
464 SCIP_VAR** vars;
465 int nvars;
466 SCIP_Real value;
467 SCIP_VAR* var;
468 SCIP_CONS* cons;
469
470 assert( scip != NULL );
471 assert( sol != NULL );
472 assert( conshdlrdata != NULL );
473
474 vars = conshdlrdata->vars;
475 nvars = conshdlrdata->nvars;
476
477 /* allocate buffer memory */
478 SCIP_CALL( SCIPallocBufferArray(scip, &consvars, nvars) );
479
480 for( v = 0; v < nvars; ++v )
481 {
482 var = vars[v];
483
484 assert( var != NULL );
485 assert( SCIPvarIsBinary(var) );
486
487 value = SCIPgetSolVal(scip, sol, var);
488 assert( SCIPisFeasIntegral(scip, value) );
489
490 if( value > 0.5 )
491 {
492 SCIP_CALL( SCIPgetNegatedVar(scip, var, &consvars[v]) );
493 }
494 else
495 consvars[v] = var;
496 }
497
498 /* create constraint */
499 SCIP_CALL( SCIPcreateConsSetcover(scip, &cons, "Setcovering created by countsols", nvars, consvars,
501
502 /* add and release constraint */
503 SCIP_CALL( SCIPaddCons(scip, cons) );
504 SCIP_CALL( SCIPreleaseCons(scip, &cons) );
505
506 /* free buffer array */
507 SCIPfreeBufferArray(scip, &consvars);
508
509 return SCIP_OKAY;
510}
511
512
513/** creates and adds a bound disjunction constraints which cuts off the current solution from the feasibility region; if
514 * only binary variables are involved, then a set covering constraint is created which is a special case of a bound
515 * disjunction constraint
516 */
517static
518CUTOFF_CONSTRAINT(addIntegerCons)
519{
520 int v;
521 SCIP_VAR** consvars;
522 SCIP_VAR** vars;
523 SCIP_Real* bounds;
524 SCIP_BOUNDTYPE* boundtypes;
525 int nvars;
526 int nbinvars = 0;
527 int nconsvars;
528 SCIP_VAR* var;
529 SCIP_Real value;
530 SCIP_CONS* cons;
531
532 assert( scip != NULL );
533 assert( sol != NULL );
534 assert( conshdlrdata != NULL );
535
536 vars = conshdlrdata->vars;
537 nvars = conshdlrdata->nvars;
538
539 nconsvars = nvars * 2;
540 assert( nvars > 0 );
541
542 /* allocate buffer memory */
543 SCIP_CALL( SCIPallocBufferArray(scip, &consvars, nconsvars) );
544 SCIP_CALL( SCIPallocBufferArray(scip, &bounds, nconsvars) );
545 SCIP_CALL( SCIPallocBufferArray(scip, &boundtypes, nconsvars) );
546
547 nconsvars = 0;
548
549 for( v = nvars - 1; v >= 0; --v )
550 {
551 var = vars[v];
552
553 assert( SCIPvarGetType(var) != SCIP_VARTYPE_CONTINUOUS );
554
555 if( SCIPvarIsBinary(var) )
556 {
557 ++nbinvars;
558 value = SCIPgetSolVal(scip, sol, var);
559 assert( SCIPisFeasIntegral(scip, value) );
560
561 if( value < 0.5 )
562 {
563 boundtypes[nconsvars] = SCIP_BOUNDTYPE_LOWER;
564 bounds[nconsvars] = 1;
565 }
566 else
567 {
568 boundtypes[nconsvars] = SCIP_BOUNDTYPE_UPPER;
569 bounds[nconsvars] = 0;
570 }
571 }
572 else
573 {
574 SCIP_Real lb;
575 SCIP_Real ub;
576 SCIP_Real valueInt;
577
580 assert( SCIPisFeasIntegral(scip, SCIPgetSolVal(scip, sol, var)) );
581
582 lb = SCIPvarGetLbLocal(var);
583 ub = SCIPvarGetUbLocal(var);
584 valueInt = SCIPgetSolVal(scip, sol, var);
585
586 if( SCIPisFeasEQ(scip, valueInt, lb) )
587 {
588 boundtypes[nconsvars] = SCIP_BOUNDTYPE_LOWER;
589 bounds[nconsvars] = lb + 1.0;
590 }
591 else if( SCIPisFeasEQ(scip, valueInt, ub) )
592 {
593 boundtypes[nconsvars] = SCIP_BOUNDTYPE_UPPER;
594 bounds[nconsvars] = ub - 1.0;
595 }
596 else
597 {
598 boundtypes[nconsvars] = SCIP_BOUNDTYPE_LOWER;
599 bounds[nconsvars] = valueInt + 1.0;
600 consvars[nconsvars] = var;
601 ++nconsvars;
602 boundtypes[nconsvars] = SCIP_BOUNDTYPE_UPPER;
603 bounds[nconsvars] = valueInt - 1.0;
604 }
605 }
606
607 consvars[nconsvars] = var;
608 ++nconsvars;
609 }
610
611 /* check if only binary variables appear in the constraint; if this is the case, we
612 * create a set covering constraint instead of a bound disjunction constraint
613 */
614 if( nvars == nbinvars )
615 {
616 for( v = nbinvars - 1; v >= 0; --v )
617 {
618 /* in the case the bound is zero we have use the negated variable */
619 if( bounds[v] == 0)
620 {
621 SCIP_CALL( SCIPgetNegatedVar(scip, consvars[v], &consvars[v]));
622 }
623 }
624
625 SCIP_CALL( SCIPcreateConsSetcover(scip, &cons, "Setcovering created by countsols", nbinvars, consvars,
627 }
628 else
629 {
630 SCIP_CALL( SCIPcreateConsBounddisjunction(scip, &cons, "Bounddisjunction created by countsols",
631 nconsvars, consvars, boundtypes, bounds,
633 }
634
635 /* add and release constraint locally */
636 SCIP_CALL( SCIPaddCons(scip, cons) );
637 SCIP_CALL( SCIPreleaseCons(scip, &cons) );
638
639 /* free buffer memory */
640 SCIPfreeBufferArray(scip, &consvars);
641 SCIPfreeBufferArray(scip, &bounds);
642 SCIPfreeBufferArray(scip, &boundtypes);
643
644 return SCIP_OKAY;
645}
646
647/** collect given solution or local domains as sparse solution */
648static
650 SCIP* scip, /**< SCIP data structure */
651 SCIP_CONSHDLRDATA* conshdlrdata, /**< constraint handler data */
652 SCIP_SOL* sol /**< solution, or NULL if local domains */
653 )
654{
655 SCIP_SPARSESOL* solution;
656 SCIP_Longint* lbvalues;
657 SCIP_Longint* ubvalues;
658 int nvars;
659 int v;
660
661 /* ensure size of solution array
662 *
663 * we use normal memory instead of block memory because this plugin is rarely used, the size of 'solutions'
664 * can be arbitrary large, and the change that the other blocks can be used is quite small
665 */
666 if( conshdlrdata->nsolutions == conshdlrdata->ssolutions )
667 {
668 if( conshdlrdata->ssolutions == 0 )
669 {
670 conshdlrdata->ssolutions = 100;
671 SCIP_CALL( SCIPallocMemoryArray(scip, &conshdlrdata->solutions, conshdlrdata->ssolutions) );
672 }
673 else
674 {
675 assert( conshdlrdata->ssolutions < INT_MAX / 2);
676 conshdlrdata->ssolutions *= 2;
677 SCIP_CALL( SCIPreallocMemoryArray(scip, &conshdlrdata->solutions, conshdlrdata->ssolutions) );
678 }
679 }
680 assert( conshdlrdata->nsolutions < conshdlrdata->ssolutions );
681
682 /* get number of active variables */
683 nvars = conshdlrdata->nvars;
684
685 SCIPdebugMsg(scip, "creating solution number %d\n", conshdlrdata->nsolutions);
686
687 /* create a solution */
688 SCIP_CALL_FINALLY( SCIPsparseSolCreate(&solution, conshdlrdata->vars, nvars, FALSE), SCIPsparseSolFree(&solution) );
689 assert(solution != NULL);
690
691 lbvalues = SCIPsparseSolGetLbs(solution);
692 ubvalues = SCIPsparseSolGetUbs(solution);
693 assert(ubvalues != NULL);
694 assert(lbvalues != NULL);
695
696 for( v = nvars - 1; v >= 0; --v )
697 {
698 SCIP_VAR* var;
699
700 var = conshdlrdata->vars[v];
701 assert(var != NULL);
702
703 if( sol == NULL )
704 {
707 }
708 else
709 {
710 lbvalues[v] = SCIPconvertRealToLongint(scip, SCIPgetSolVal(scip, sol, var));
711 ubvalues[v] = lbvalues[v];
712 }
713
714 SCIPdebugMsg(scip, "variable <%s> [%" SCIP_LONGINT_FORMAT ",%" SCIP_LONGINT_FORMAT "]\n",
715 SCIPvarGetName(var), lbvalues[v], ubvalues[v]);
716 }
717
718 conshdlrdata->solutions[conshdlrdata->nsolutions] = solution;
719 conshdlrdata->nsolutions++;
720
721 return SCIP_OKAY;
722}
723
724
725/** counts the number of solutions represented by sol */
726static
728 SCIP* scip, /**< SCIP data structure */
729 SCIP_SOL* sol, /**< solution */
730 SCIP_Bool feasible, /**< is solution feasible? */
731 SCIP_CONSHDLRDATA* conshdlrdata, /**< constraint handler data */
732 SCIP_RESULT* result /**< pointer to store the result of the checking process */
733 )
734{
735 assert( scip != NULL );
736 assert( sol != NULL );
737 assert( conshdlrdata != NULL );
738 assert( result != NULL );
739
740 /* the result should be infeasible since we reject any solution; however, if the solution passes the sparse test, the
741 * result is set to SCIP_CUTOFF which cuts off the subtree initialized through the current node
742 */
743 assert(*result == SCIP_INFEASIBLE);
744
745 if( feasible )
746 {
747 int v;
748 Int newsols;
749 SCIP_VAR** vars;
750 int nvars;
751 SCIP_VAR* var;
752 SCIP_Real lb;
753 SCIP_Real ub;
754
755 SCIPdebugMsg(scip, "counts number of solutions represented through the given one\n");
756
757 /**@note aggregations and multi aggregations: we do not have to care about these things
758 * since we count solutions from the transformed problem and therefore, SCIP does
759 * it for us
760 */
761 assert( SCIPgetNPseudoBranchCands(scip) != 0 );
762
763 allocInt(&newsols); /*lint !e545*/
764
765 /* set newsols to one */
766 setInt(&newsols, 1LL); /*lint !e545*/
767
769 {
770 int npseudocands;
771
772 npseudocands = SCIPgetNPseudoBranchCands(scip);
773
774 /* sets a power of 2 to the number of solutions */
775 setPowerOfTwo(&newsols, (SCIP_Longint) npseudocands); /*lint !e545*/
776 }
777 else
778 {
779 SCIP_VAR* origvar;
780 SCIP_Real scalar = 1.0;
781 SCIP_Real constant = 0.0;
782
783 SCIP_CALL( SCIPgetPseudoBranchCands(scip, &vars, &nvars, NULL) );
784
785 for( v = 0; v < nvars; ++v )
786 {
787 var = vars[v];
788 origvar = var;
789
790 /* get original variable to decide if we will count the domain; continuous variables aren't counted */
791 SCIP_CALL( SCIPvarGetOrigvarSum(&origvar, &scalar, &constant) );
792
793 if( origvar != NULL && SCIPvarGetType(origvar) != SCIP_VARTYPE_CONTINUOUS )
794 {
795 lb = SCIPvarGetLbLocal(var);
796 ub = SCIPvarGetUbLocal(var);
797
798 SCIPdebugMsg(scip, "variable <%s> Local Bounds are [%g,%g]\n", SCIPvarGetName(var), lb, ub);
799
800 assert( SCIPvarGetType(var) != SCIP_VARTYPE_CONTINUOUS );
801 assert( SCIPisFeasIntegral(scip, lb) );
802 assert( SCIPisFeasIntegral(scip, ub) );
803 assert( SCIPisFeasIntegral(scip, ub - lb) );
804 assert( SCIPisFeasLT(scip, lb, ub) );
805
806 /* the number of integers lying in the interval [lb,ub] is (ub - lb + 1); to make everything integral we
807 * add another 0.5 and cut the fractional part off
808 */
809 multInt(&newsols, (SCIP_Longint)(ub - lb + 1.5) ); /*lint !e545*/
810 }
811 }
812 }
813
814 *result = SCIP_CUTOFF;
815 conshdlrdata->feasST++;
816
817 if( conshdlrdata->collect )
818 {
819 SCIP_CALL( collectSolution(scip, conshdlrdata, NULL) );
820 }
821
822 addInt(&conshdlrdata->nsols, &newsols); /*lint !e545*/
823 freeInt(&newsols); /*lint !e545*/
824 }
825 else if(!conshdlrdata->discardsols)
826 {
827 SCIP_CALL( conshdlrdata->cutoffSolution(scip, sol, conshdlrdata) );
828 addOne(&conshdlrdata->nsols); /*lint !e545*/
829 conshdlrdata->nNonSparseSols++;
830 if( conshdlrdata->collect )
831 {
832 SCIP_CALL( collectSolution(scip, conshdlrdata, sol) );
833 }
834 }
835 else
836 conshdlrdata->nDiscardSols++;
837
838 return SCIP_OKAY;
839}
840
841
842/** checks if the new solution is feasible for the logicor constraints */
843static
845 SCIP* scip, /**< SCIP data structure */
846 SCIP_CONSHDLR* conshdlr, /**< constraint handler */
847 int nconss, /**< number of enabled constraints */
848 SCIP_Bool* satisfied /**< pointer to store if the logicor constraints a satisfied */
849 )
850{
851 /**@note the logicor constraints are not fully propagated; therefore, we have to check
852 * them by hand if they are satisfied or not; if a constraint is satisfied we
853 * delete it locally from the branch and bound tree.
854 */
855
856 SCIP_CONS** conss;
857 SCIP_VAR** vars;
858 SCIP_Bool fixedone;
859 int nvars;
860 int c;
861 int v;
862
863 SCIPdebugMsg(scip, "check logicor %d constraints\n", nconss);
864
865 assert( scip != NULL );
866 assert( conshdlr != NULL );
867 assert( strcmp(SCIPconshdlrGetName(conshdlr),"logicor") == 0 );
868 assert( nconss == SCIPconshdlrGetNEnabledConss(conshdlr) );
869
870 conss = SCIPconshdlrGetConss(conshdlr);
871 assert( conss != NULL );
872
873 (*satisfied) = TRUE;
874 c = SCIPconshdlrGetNActiveConss(conshdlr) - 1;
875
876 for( ; c >= 0 && nconss > 0 && (*satisfied); --c )
877 {
878 SCIPdebugMsg(scip, "logicor constraint %d\n", c);
879
880 if( !SCIPconsIsEnabled(conss[c]) )
881 continue;
882
883 nconss--;
884
885 nvars = SCIPgetNVarsLogicor(scip, conss[c]);
886 vars = SCIPgetVarsLogicor(scip, conss[c]);
887
888 /* calculate the constraint's activity */
889 fixedone = FALSE;
890 for( v = 0; v < nvars && !fixedone; ++v )
891 {
892 assert(SCIPvarIsBinary(vars[v]));
893
894 if( !varIsUnfixedLocal(vars[v] ) )
895 fixedone = SCIPvarGetLbLocal(vars[v]) > 0.5;
896 }
897
898 if( !fixedone )
899 {
900 SCIPdebugMsg(scip, "constraint <%s> cannot be disabled\n", SCIPconsGetName(conss[c]));
901 SCIPdebugPrintCons(scip, conss[c], NULL);
902 (*satisfied) = FALSE;
903 }
904 else
905 {
906 /* delete constraint from the problem locally since it is satisfied */
907 SCIP_CALL( SCIPdelConsLocal(scip, conss[c]) );
908 }
909 }
910
911 return SCIP_OKAY;
912}
913
914
915/** checks if the new solution is feasible for the knapsack constraints */
916static
918 SCIP* scip, /**< SCIP data structure */
919 SCIP_CONSHDLR* conshdlr, /**< constraint handler */
920 int nconss, /**< number of enabled constraints */
921 SCIP_Bool* satisfied /**< pointer to store if the logicor constraints a satisfied */
922 )
923{
924 /**@note the knapsack constraints are not fully propagated; therefore, we have to check
925 * them by hand if they are satisfied or not; if a constraint is satisfied we
926 * delete it locally from the branch and bound tree.
927 */
928
929 SCIP_CONS** conss;
930 SCIP_VAR** vars;
931 SCIP_Longint* weights;
932 SCIP_Longint capacity;
933 SCIP_Real capa;
934 int nvars;
935 int c;
936 int v;
937
938 SCIPdebugMsg(scip, "check knapsack %d constraints\n", nconss);
939
940 assert( scip != NULL );
941 assert( conshdlr != NULL );
942 assert( strcmp(SCIPconshdlrGetName(conshdlr),"knapsack") == 0 );
943 assert( nconss == SCIPconshdlrGetNEnabledConss(conshdlr) );
944
945 conss = SCIPconshdlrGetConss(conshdlr);
946 assert( conss != NULL );
947
948 (*satisfied) = TRUE;
949 c = SCIPconshdlrGetNActiveConss(conshdlr) - 1;
950
951 for( ; c >= 0 && nconss > 0 && (*satisfied); --c )
952 {
953 SCIPdebugMsg(scip, "knapsack constraint %d\n", c);
954
955 if( !SCIPconsIsEnabled(conss[c]) )
956 continue;
957
958 nconss--;
959
960 nvars = SCIPgetNVarsKnapsack(scip, conss[c]);
961 vars = SCIPgetVarsKnapsack(scip, conss[c]);
962 capacity = SCIPgetCapacityKnapsack(scip, conss[c]);
963 weights = SCIPgetWeightsKnapsack(scip,conss[c]);
964
965 SCIPdebugMsg(scip, "knapsack capacity = %" SCIP_LONGINT_FORMAT "\n", capacity);
966
967 capa = capacity + 0.1;
968
969 for( v = nvars - 1; v >= 0 && capa >= 0 ; --v )
970 {
971 SCIPdebug( SCIP_CALL( SCIPprintVar( scip, vars[v], NULL) ) );
972 SCIPdebugMsg(scip, "weight = %" SCIP_LONGINT_FORMAT " :\n", weights[v]);
973 assert( SCIPvarIsIntegral(vars[v]) );
974
975 /* the weights should be greater or equal to zero */
976 assert( weights[v] >= 0);
977
978 if( !varIsUnfixedLocal(vars[v]) )
979 {
980 /* variable is fixed locally; therefore, subtract fixed variable value multiplied by
981 * the weight;
982 */
983 capa -= weights[v] * SCIPvarGetLbLocal(vars[v]);
984 }
985 else if( weights[v] >= 1 )
986 {
987 /* variable is unfixed and weight is greater than 0; therefore, subtract upper bound
988 * value multiplied by the weight
989 */
990 capa -= weights[v] * SCIPvarGetUbLocal(vars[v]);
991 }
992 }
993
994 if( SCIPisFeasLT(scip, capa, 0.0) )
995 {
996 SCIPdebugMsg(scip, "constraint %s cannot be disabled\n", SCIPconsGetName(conss[c]));
997 SCIPdebugPrintCons(scip, conss[c], NULL);
998 (*satisfied) = FALSE;
999 }
1000 else
1001 {
1002 /* delete constraint from the problem locally since it is satisfied */
1003 SCIP_CALL( SCIPdelConsLocal(scip, conss[c]) );
1004 }
1005 }
1006 return SCIP_OKAY;
1007}
1008
1009
1010/** checks if the new solution is feasible for the bounddisjunction constraints */
1011static
1013 SCIP* scip, /**< SCIP data structure */
1014 SCIP_CONSHDLR* conshdlr, /**< constraint handler */
1015 int nconss, /**< number of enabled constraints */
1016 SCIP_Bool* satisfied /**< pointer to store if the logicor constraints a satisfied */
1017 )
1018{
1019 /**@note the bounddisjunction constraints are not fully propagated; therefore, we have to check
1020 * them by hand if they are satisfied or not; if a constraint is satisfied we
1021 * delete it locally from the branch and bound tree
1022 */
1023
1024 SCIP_CONS** conss;
1025 SCIP_VAR** vars;
1026 SCIP_BOUNDTYPE* boundtypes;
1027 SCIP_Real* bounds;
1028 SCIP_Bool satisfiedbound;
1029 int nvars;
1030 int c;
1031 int v;
1032
1033 assert( scip != NULL );
1034 assert( conshdlr != NULL );
1035 assert( strcmp(SCIPconshdlrGetName(conshdlr),"bounddisjunction") == 0 );
1036 assert( nconss == SCIPconshdlrGetNEnabledConss(conshdlr) );
1037
1038 conss = SCIPconshdlrGetConss(conshdlr);
1039 assert( conss != NULL );
1040
1041 (*satisfied) = TRUE;
1042 c = SCIPconshdlrGetNActiveConss(conshdlr) - 1;
1043
1044 for( ; c >= 0 && nconss > 0 && (*satisfied); --c )
1045 {
1046 if( !SCIPconsIsEnabled(conss[c]) )
1047 continue;
1048
1049 nconss--;
1050 satisfiedbound = FALSE;
1051
1052 nvars = SCIPgetNVarsBounddisjunction(scip, conss[c]);
1053 vars = SCIPgetVarsBounddisjunction(scip, conss[c]);
1054
1055 boundtypes = SCIPgetBoundtypesBounddisjunction(scip, conss[c]);
1056 bounds = SCIPgetBoundsBounddisjunction(scip, conss[c]);
1057
1058 for( v = nvars-1; v >= 0 && !satisfiedbound; --v )
1059 {
1060 SCIPdebug( SCIPprintVar(scip, vars[v], NULL) );
1061
1062 /* variable should be in right bounds to delete constraint */
1063 if( boundtypes[v] == SCIP_BOUNDTYPE_LOWER )
1064 satisfiedbound = SCIPisFeasGE(scip, SCIPvarGetLbLocal(vars[v]), bounds[v]);
1065 else
1066 {
1067 assert( boundtypes[v] == SCIP_BOUNDTYPE_UPPER );
1068 satisfiedbound = SCIPisFeasLE(scip, SCIPvarGetUbLocal(vars[v]), bounds[v]);
1069 }
1070 }
1071
1072 if( !satisfiedbound )
1073 {
1074 SCIPdebugMsg(scip, "constraint %s cannot be disabled\n", SCIPconsGetName(conss[c]));
1075 SCIPdebugPrintCons(scip, conss[c], NULL);
1076 (*satisfied) = FALSE;
1077 }
1078 else
1079 {
1080 /* delete constraint from the problem locally since it is satisfied */
1081 SCIP_CALL( SCIPdelConsLocal(scip, conss[c]) );
1082 }
1083 }
1084 return SCIP_OKAY;
1085}
1086
1087
1088/** checks if the new solution is feasible for the varbound constraints */
1089static
1091 SCIP* scip, /**< SCIP data structure */
1092 SCIP_CONSHDLR* conshdlr, /**< constraint handler */
1093 int nconss, /**< number of enabled constraints */
1094 SCIP_Bool* satisfied /**< pointer to store if the logicor constraints a satisfied */
1095 )
1096{
1097 /**@note the varbound constraints are not fully propagated; therefore, we have to check
1098 * them by hand if they are satisfied or not; if a constraint is satisfied we
1099 * delete it locally from the branch and bound tree.
1100 */
1101
1102 SCIP_CONS** conss;
1103 SCIP_VAR* var;
1104 SCIP_VAR* vbdvar;
1105 SCIP_Real lhs;
1106 SCIP_Real rhs;
1107 SCIP_Real coef;
1108 int c;
1109
1110 SCIPdebugMsg(scip, "check varbound %d constraints\n", nconss);
1111
1112 assert( scip != NULL );
1113 assert( conshdlr != NULL );
1114 assert( strcmp(SCIPconshdlrGetName(conshdlr),"varbound") == 0 );
1115 assert( nconss == SCIPconshdlrGetNEnabledConss(conshdlr) );
1116
1117 conss = SCIPconshdlrGetConss(conshdlr);
1118 assert( conss != NULL );
1119
1120 (*satisfied) = TRUE;
1121 c = SCIPconshdlrGetNActiveConss(conshdlr) - 1;
1122
1123 for( ; c >= 0 && nconss > 0 && (*satisfied); --c )
1124 {
1125 SCIPdebugMsg(scip, "varbound constraint %d\n", c);
1126
1127 if( !SCIPconsIsEnabled(conss[c]) )
1128 continue;
1129
1130 nconss--;
1131
1132 var = SCIPgetVarVarbound(scip, conss[c]);
1133 vbdvar = SCIPgetVbdvarVarbound(scip, conss[c]);
1134
1135 assert (SCIPvarGetType(vbdvar) != SCIP_VARTYPE_CONTINUOUS);
1136
1137 coef = SCIPgetVbdcoefVarbound(scip, conss[c]);
1138 lhs = SCIPgetLhsVarbound(scip, conss[c]);
1139 rhs = SCIPgetRhsVarbound(scip, conss[c]);
1140
1141 /* variables y is fixed locally; therefore, subtract fixed variable value multiplied by
1142 * the coefficient;
1143 */
1144 if(SCIPisGT(scip, SCIPvarGetUbLocal(var), rhs - SCIPvarGetUbLocal(vbdvar) * coef )
1145 || !SCIPisGE(scip, SCIPvarGetLbLocal(var), lhs - SCIPvarGetLbLocal(vbdvar) * coef ) )
1146 {
1147 SCIPdebugMsg(scip, "constraint %s cannot be disabled\n", SCIPconsGetName(conss[c]));
1148 SCIPdebugPrintCons(scip, conss[c], NULL);
1149 SCIPdebugMsg(scip, "<%s> lb: %.15g\t ub: %.15g\n", SCIPvarGetName(var), SCIPvarGetLbLocal(var), SCIPvarGetUbLocal(var));
1150 SCIPdebugMsg(scip, "<%s> lb: %.15g\t ub: %.15g\n", SCIPvarGetName(vbdvar), SCIPvarGetLbLocal(vbdvar), SCIPvarGetUbLocal(vbdvar));
1151 (*satisfied) = FALSE;
1152 }
1153 else
1154 {
1155 /* delete constraint from the problem locally since it is satisfied */
1156 SCIP_CALL( SCIPdelConsLocal(scip, conss[c]) );
1157 }
1158 }
1159
1160 return SCIP_OKAY;
1161}
1162
1163
1164/** check if the current node initializes a non trivial unrestricted subtree */
1165static
1167 SCIP* scip, /**< SCIP main data structure */
1168 SCIP_SOL* sol, /**< solution to check */
1169 SCIP_Bool* feasible /**< pointer to store the result of the check */
1170 )
1171{
1172 int h;
1173
1174 SCIP_CONSHDLR** conshdlrs;
1175 int nconshdlrs;
1176
1177 SCIP_CONSHDLR* conshdlr;
1178 int nconss;
1179
1180 SCIPdebugMsg(scip, "check if the sparse solution is feasible\n");
1181
1182 assert( scip != NULL );
1183 assert( sol != NULL );
1184 assert( feasible != NULL );
1185
1186 assert( SCIPgetNPseudoBranchCands(scip) != 0 );
1187
1188 *feasible = FALSE;
1189
1190 nconshdlrs = SCIPgetNConshdlrs(scip) - 1;
1191 conshdlrs = SCIPgetConshdlrs(scip);
1192 assert(conshdlrs != NULL);
1193
1194 /* check each constraint handler if there are constraints which are not enabled */
1195 for( h = nconshdlrs ; h >= 0 ; --h )
1196 {
1197 conshdlr = conshdlrs[h];
1198 assert( conshdlr != NULL );
1199
1200 /* skip this constraints handler */
1201 if( strcmp(SCIPconshdlrGetName(conshdlr), CONSHDLR_NAME) == 0 )
1202 continue;
1203
1204 nconss = SCIPconshdlrGetNEnabledConss(conshdlr);
1205
1206 if( nconss > 0 )
1207 {
1208 SCIP_Bool satisfied;
1209
1210 SCIPdebugMsg(scip, "constraint handler %s has %d active constraint(s)\n",
1211 SCIPconshdlrGetName(conshdlr), nconss );
1212
1213 if( strcmp(SCIPconshdlrGetName(conshdlr), "logicor") == 0 )
1214 {
1215 SCIP_CALL( checkLogicor(scip, conshdlr, nconss, &satisfied) );
1216 if( !satisfied )
1217 {
1218 SCIPdebugMsg(scip, "a <logicor> constraint cannot be disabled\n");
1219 return SCIP_OKAY;
1220 }
1221 }
1222 else if( strcmp(SCIPconshdlrGetName(conshdlr), "knapsack") == 0 )
1223 {
1224 SCIP_CALL( checkKnapsack(scip, conshdlr, nconss, &satisfied) );
1225 if( !satisfied )
1226 {
1227 SCIPdebugMsg(scip, "a <knapsack> constraint cannot be disabled\n");
1228 return SCIP_OKAY;
1229 }
1230 }
1231 else if( strcmp(SCIPconshdlrGetName(conshdlr), "bounddisjunction") == 0 )
1232 {
1233 SCIP_CALL( checkBounddisjunction(scip, conshdlr, nconss, &satisfied) );
1234 if( !satisfied )
1235 {
1236 SCIPdebugMsg(scip, "a <bounddisjunction> constraint cannot be disabled\n");
1237 return SCIP_OKAY;
1238 }
1239 }
1240 else if( strcmp(SCIPconshdlrGetName(conshdlr), "varbound") == 0 )
1241 {
1242 SCIP_CALL( checkVarbound(scip, conshdlr, nconss, &satisfied) );
1243 if( !satisfied )
1244 {
1245 SCIPdebugMsg(scip, "a <varbound> constraint cannot be disabled\n");
1246 return SCIP_OKAY;
1247 }
1248 }
1249 else
1250 {
1251 SCIPdebugMsg(scip, "sparse solution is infeasible since the following constraint (and maybe more) is(/are) enabled\n");
1253 return SCIP_OKAY;
1254 }
1255 }
1256 }
1257
1258 *feasible = TRUE;
1259 SCIPdebugMsg(scip, "sparse solution is feasible\n");
1260
1261 return SCIP_OKAY;
1262}
1263
1264
1265/** check the given solution */
1266static
1268 SCIP* scip, /**< SCIP data structure */
1269 SCIP_SOL* sol, /**< solution to add */
1270 SCIP_CONSHDLRDATA* conshdlrdata, /**< constraint handler data */
1271 SCIP_RESULT* result /**< pointer to store the result of the checking process */
1272 )
1273{
1274 SCIP_Longint nsols;
1275 SCIP_Bool feasible;
1276 SCIP_Bool valid;
1277
1278 SCIPdebugMsg(scip, "start to add sparse solution\n");
1279
1280 assert( scip != NULL );
1281 assert( sol != NULL );
1282 assert( conshdlrdata != NULL );
1283 assert( result != NULL );
1284
1285 /* the solution should not be found through a heuristic since in this case the information of SCIP is not valid for
1286 * this solution
1287 */
1288
1289 /**@todo it might be not necessary to check this assert since we can check in general all solutions of feasibility
1290 * independently of the origin; however, the locally fixed technique does only work if the solution comes from
1291 * the branch and bound tree; in case the solution comes from a heuristic we should try to sequentially fix the
1292 * variables in the branch and bound tree and check after every fixing if all constraints are disabled; at the
1293 * point where all constraints are disabled the unfixed variables are "stars" (arbitrary);
1294 */
1295 assert( SCIPsolGetHeur(sol) == NULL);
1296
1297 /* setting result to infeasible since we reject any solution; however, if the solution passes the sparse test or is
1298 * completely fixed, the result is set to SCIP_CUTOFF which cuts off the subtree initialized through the current node
1299 */
1300 *result = SCIP_INFEASIBLE;
1301
1302#ifdef SCIP_DEBUG
1303 {
1304 SCIP_VAR* var;
1305 SCIP_VAR** vars;
1306 int v;
1307 int nvars;
1308
1309 nvars = SCIPgetNVars(scip);
1310 vars = SCIPgetVars(scip);
1311
1312 for( v = 0; v < nvars; ++v )
1313 {
1314 var = vars[v];
1315 SCIPdebugMsg(scip, "variables <%s> Local Bounds are [%g,%g] Global Bounds are [%g,%g]\n",
1317 }
1318 }
1319#endif
1320
1321 /* check if integer variables are completely fixed */
1323 {
1324 /* check solution original space */
1325 checkSolutionOrig(scip, sol, conshdlrdata);
1326
1327 addOne(&conshdlrdata->nsols); /*lint !e545*/
1328 conshdlrdata->nNonSparseSols++;
1329
1330 SCIPdebugMsg(scip, "-> add one to number of solutions\n");
1331
1332 if( conshdlrdata->collect )
1333 {
1334 SCIP_CALL( collectSolution(scip, conshdlrdata, sol) );
1335 }
1336
1337 /* in case of continuous variables are present we explicitly cutoff the integer assignment since in case of
1338 * nonlinear constraint we want to avoid to count that integer assignment again
1339 */
1340 if( conshdlrdata->continuous )
1341 {
1342 SCIP_CALL( conshdlrdata->cutoffSolution(scip, sol, conshdlrdata) );
1343 }
1344
1345 /* since all integer are fixed, we cut off the subtree */
1346 *result = SCIP_CUTOFF;
1347 }
1348 else if( conshdlrdata->sparsetest )
1349 {
1350 SCIP_CALL( checkFeasSubtree(scip, sol, &feasible) ) ;
1351 SCIP_CALL( countSparseSol(scip, sol, feasible, conshdlrdata, result) );
1352 }
1353
1354 /* transform the current number of solutions into a SCIP_Longint */
1355 nsols = getNCountedSols(conshdlrdata->nsols, &valid);
1356
1357 /* check if the solution limit is hit and stop SCIP if this is the case */
1358 if( conshdlrdata->sollimit > -1 && (!valid || conshdlrdata->sollimit <= nsols) )
1359 {
1361 }
1362
1363 assert( *result == SCIP_INFEASIBLE || *result == SCIP_CUTOFF );
1364 SCIPdebugMsg(scip, "result is %s\n", *result == SCIP_INFEASIBLE ? "SCIP_INFEASIBLE" : "SCIP_CUTOFF" );
1365
1366 return SCIP_OKAY;
1367}
1368
1369/*
1370 * Callback methods of constraint handler
1371 */
1372
1373/** creates the handler for countsols constraints and includes it in SCIP */
1374static
1376 SCIP* scip, /**< SCIP data structure */
1377 SCIP_Bool dialogs /**< sould count dialogs be added */
1378 );
1379
1380/** copy method for constraint handler plugins (called when SCIP copies plugins) */
1381static
1382SCIP_DECL_CONSHDLRCOPY(conshdlrCopyCountsols)
1383{ /*lint --e{715}*/
1384 SCIP_CONSHDLRDATA* conshdlrdata;
1385
1386 conshdlrdata = SCIPconshdlrGetData(conshdlr);
1387 assert(conshdlrdata != NULL);
1388
1389 /* in case the countsols constraint handler is active we avoid copying to ensure a safe count */
1390 if( conshdlrdata->active )
1391 *valid = FALSE;
1392 else
1393 {
1394 assert(scip != NULL);
1395 assert(conshdlr != NULL);
1396 assert(strcmp(SCIPconshdlrGetName(conshdlr), CONSHDLR_NAME) == 0);
1397
1398 /* call inclusion method of constraint handler and do not add counting dialogs */
1400
1401 *valid = TRUE;
1402 }
1403
1404 return SCIP_OKAY;
1405}
1406
1407#define consCopyCountsols NULL
1408
1409/** destructor of constraint handler to free constraint handler data (called when SCIP is exiting) */
1410static
1411SCIP_DECL_CONSFREE(consFreeCountsols)
1412{ /*lint --e{715}*/
1413 SCIP_CONSHDLRDATA* conshdlrdata;
1414
1415 assert(conshdlr != NULL);
1416 assert(strcmp(SCIPconshdlrGetName(conshdlr), CONSHDLR_NAME) == 0);
1417
1418 /* free constraint handler data */
1419 conshdlrdata = SCIPconshdlrGetData(conshdlr);
1420 assert(conshdlrdata != NULL);
1421
1422 /* free conshdlrdata */
1423 freeInt(&conshdlrdata->nsols); /*lint !e545*/
1424
1425 assert( conshdlrdata->solutions == NULL );
1426 assert( conshdlrdata->nsolutions == 0 );
1427 assert( conshdlrdata->ssolutions == 0 );
1428
1429 SCIPfreeBlockMemory(scip, &conshdlrdata);
1430 SCIPconshdlrSetData(conshdlr, NULL);
1431
1432 return SCIP_OKAY;
1433}
1434
1435/** initialization method of constraint handler (called after problem was transformed) */
1436static
1437SCIP_DECL_CONSINIT(consInitCountsols)
1438{ /*lint --e{715}*/
1439 SCIP_CONSHDLRDATA* conshdlrdata;
1440
1441 assert( conshdlr != NULL );
1442 assert( strcmp(SCIPconshdlrGetName(conshdlr), CONSHDLR_NAME) == 0 );
1443
1444 conshdlrdata = SCIPconshdlrGetData(conshdlr);
1445 assert(conshdlrdata != NULL );
1446
1447 /* reset counting variables */
1448 conshdlrdata->feasST = 0; /* number of non trivial unrestricted subtrees */
1449 conshdlrdata->nDiscardSols = 0; /* number of discard solutions */
1450 conshdlrdata->nNonSparseSols = 0; /* number of non sparse solutions */
1451 setInt(&conshdlrdata->nsols, 0LL); /* number of solutions */ /*lint !e545*/
1452
1453 conshdlrdata->solutions = NULL;
1454 conshdlrdata->nsolutions = 0;
1455 conshdlrdata->ssolutions = 0;
1456
1457 if( conshdlrdata->active )
1458 {
1459 SCIP_VAR** origvars;
1460 int norigvars;
1461 int nallvars;
1462 int v;
1463
1464 origvars = SCIPgetOrigVars(scip);
1465 norigvars = SCIPgetNOrigVars(scip);
1466
1467 /* get number of integral variables */
1468 conshdlrdata->nallvars = SCIPgetNVars(scip) - SCIPgetNContVars(scip);
1469
1470 SCIP_CALL( SCIPallocBlockMemoryArray(scip, &conshdlrdata->allvars, conshdlrdata->nallvars) );
1471
1472 nallvars = 0;
1473
1474 /* capture and lock all variables */
1475 for( v = 0; v < norigvars; ++v )
1476 {
1477 if( SCIPvarGetType(origvars[v]) != SCIP_VARTYPE_CONTINUOUS )
1478 {
1479 assert(nallvars < conshdlrdata->nallvars);
1480
1481 SCIP_CALL( SCIPgetTransformedVar(scip, origvars[v], &conshdlrdata->allvars[nallvars]) );
1482 assert(conshdlrdata->allvars[nallvars] != NULL);
1483
1484 /* capture variable to ensure that the variable will not be deleted */
1485 SCIP_CALL( SCIPcaptureVar(scip, conshdlrdata->allvars[nallvars]) );
1486
1487 if( strncmp(SCIPvarGetName(conshdlrdata->allvars[nallvars]), "t_andresultant_", strlen("t_andresultant_")) != 0 )
1488 {
1489 /* lock variable to avoid dual reductions */
1490 SCIP_CALL( SCIPaddVarLocksType(scip, conshdlrdata->allvars[nallvars], SCIP_LOCKTYPE_MODEL, 1, 1) );
1491 }
1492
1493 nallvars++;
1494 }
1495 }
1496 assert(nallvars == conshdlrdata->nallvars);
1497
1498 /* check if continuous variables are present */
1499 conshdlrdata->continuous = SCIPgetNContVars(scip) > 0;
1500 }
1501
1502 return SCIP_OKAY;
1503}
1504
1505/** deinitialization method of constraint handler (called before transformed problem is freed) */
1506static
1507SCIP_DECL_CONSEXIT(consExitCountsols)
1508{ /*lint --e{715}*/
1509 SCIP_CONSHDLRDATA* conshdlrdata;
1510 int s;
1511 int v;
1512
1513 assert( conshdlr != NULL );
1514 assert( strcmp(SCIPconshdlrGetName(conshdlr), CONSHDLR_NAME) == 0 );
1515
1516 conshdlrdata = SCIPconshdlrGetData(conshdlr);
1517 assert(conshdlrdata != NULL );
1518
1519 /* release variables to hashmap */
1520 for( v = conshdlrdata->nvars - 1; v >= 0; --v )
1521 {
1522 SCIP_CALL( SCIPreleaseVar(scip, &(conshdlrdata->vars[v])) );
1523 }
1524
1525 if( conshdlrdata->hashmap != NULL)
1526 {
1527 /* free hashmap of active variables to pistions */
1528 SCIPhashmapFree(&(conshdlrdata->hashmap));
1529 }
1530
1531 /* free active variables */
1532 SCIPfreeBlockMemoryArrayNull(scip, &(conshdlrdata->vars), conshdlrdata->nvars);
1533 conshdlrdata->nvars = 0;
1534
1535 if( conshdlrdata->allvars != NULL )
1536 {
1537 /* release and unlock all variables */
1538 for( v = 0; v < conshdlrdata->nallvars; ++v )
1539 {
1540 if( strncmp(SCIPvarGetName(conshdlrdata->allvars[v]), "t_andresultant_", strlen("t_andresultant_")) != 0 )
1541 {
1542 /* remove the previously added variable locks */
1543 SCIP_CALL( SCIPaddVarLocksType(scip, conshdlrdata->allvars[v], SCIP_LOCKTYPE_MODEL, -1, -1) );
1544 }
1545
1546 SCIP_CALL( SCIPreleaseVar(scip, &conshdlrdata->allvars[v]) );
1547 }
1548
1549 SCIPfreeBlockMemoryArrayNull(scip, &conshdlrdata->allvars, conshdlrdata->nallvars);
1550 conshdlrdata->nallvars = 0;
1551 }
1552
1553 if( conshdlrdata->nsolutions > 0 )
1554 {
1555 for( s = conshdlrdata->nsolutions - 1; s >= 0 ; --s )
1556 {
1557 SCIPsparseSolFree(&(conshdlrdata->solutions[s]));
1558 }
1559
1560 SCIPfreeMemoryArrayNull(scip, &conshdlrdata->solutions);
1561 conshdlrdata->nsolutions = 0;
1562 conshdlrdata->ssolutions = 0;
1563
1564 assert( conshdlrdata->solutions == NULL );
1565 }
1566 conshdlrdata->continuous = FALSE;
1567
1568 assert( conshdlrdata->solutions == NULL );
1569 assert( conshdlrdata->nsolutions == 0 );
1570 assert( conshdlrdata->ssolutions == 0 );
1571
1572 return SCIP_OKAY;
1573}
1574
1575
1576/** solving process initialization method of constraint handler (called when branch and bound process is about to begin)
1577 *
1578 * This method is called when the presolving was finished and the branch and bound process is about to begin.
1579 * The constraint handler may use this call to initialize its branch and bound specific data.
1580 */
1581static
1582SCIP_DECL_CONSINITSOL(consInitsolCountsols)
1583{ /*lint --e{715}*/
1584 SCIP_CONSHDLRDATA* conshdlrdata;
1585
1586 assert( conshdlr != NULL );
1587 assert( strcmp(SCIPconshdlrGetName(conshdlr), CONSHDLR_NAME) == 0 );
1588
1589 conshdlrdata = SCIPconshdlrGetData(conshdlr);
1590 assert(conshdlrdata != NULL );
1591
1592 if( conshdlrdata->active && SCIPgetNVars(scip) >= 1 )
1593 {
1594 SCIP_VAR** vars;
1595 int v;
1596
1597 assert(conshdlrdata->nsolutions == 0);
1598 assert(conshdlrdata->solutions == NULL);
1599
1600 conshdlrdata->nvars = SCIPgetNVars(scip) - SCIPgetNContVars(scip);
1601 vars = SCIPgetVars(scip);
1602
1603 /* exclude upgrade continuous original variables */
1604 for( v = conshdlrdata->nvars - 1; v >= 0; --v )
1605 {
1606 SCIP_VAR* origvar;
1607 SCIP_Real scalar = 1.0;
1608 SCIP_Real constant = 0.0;
1609
1610 origvar = vars[v];
1611
1612 /* get original variable to decide if we will count the domain; continuous variables aren't counted */
1613 SCIP_CALL( SCIPvarGetOrigvarSum(&origvar, &scalar, &constant) );
1614
1615 if( origvar != NULL && SCIPvarGetType(origvar) != SCIP_VARTYPE_CONTINUOUS )
1616 break;
1617 }
1618 conshdlrdata->nvars = v + 1;
1619
1620 /* @todo we need to forbid variable downgrading, from integer type to implicit integer type, e.g. done in
1621 * cons_linear
1622 */
1623#ifndef NDEBUG
1624 for( v = conshdlrdata->nvars - 1; v >= 0; --v )
1625 {
1626 SCIP_VAR* origvar;
1627 SCIP_Real scalar = 1.0;
1628 SCIP_Real constant = 0.0;
1629
1630 origvar = vars[v];
1631
1632 /* get original variable to decide if we will count the domain; continuous variables aren't counted */
1633 SCIP_CALL( SCIPvarGetOrigvarSum(&origvar, &scalar, &constant) );
1634
1635 assert(origvar != NULL && SCIPvarGetType(origvar) != SCIP_VARTYPE_CONTINUOUS);
1636 }
1637#endif
1638
1639 /* copy array of active variables */
1640 SCIP_CALL( SCIPduplicateBlockMemoryArray(scip, &(conshdlrdata->vars), vars, conshdlrdata->nvars) );
1641
1642 /* store mapping from all active variables to their position afetr presolving because during solving new variables
1643 * might be added and therefore could destroy writing collected solutions
1644 */
1645 SCIP_CALL( SCIPhashmapCreate(&(conshdlrdata->hashmap), SCIPblkmem(scip), conshdlrdata->nvars + 1) );
1646
1647 /* add variables to hashmap */
1648 for( v = conshdlrdata->nvars - 1; v >= 0; --v )
1649 {
1650 assert(SCIPvarGetProbindex(conshdlrdata->vars[v]) == v);
1651 SCIP_CALL( SCIPhashmapInsertInt(conshdlrdata->hashmap, conshdlrdata->vars[v], v+1) );
1652 SCIP_CALL( SCIPcaptureVar(scip, conshdlrdata->vars[v]) );
1653 }
1654
1655 /* check if the problem is binary (ignoring continuous variables) */
1657 conshdlrdata->cutoffSolution = addBinaryCons;
1658 else
1659 conshdlrdata->cutoffSolution = addIntegerCons;
1660 }
1661
1662 return SCIP_OKAY;
1663}
1664
1665/** solving process deinitialization method of constraint handler (called before branch and bound process data is freed) */
1666static
1667SCIP_DECL_CONSEXITSOL(consExitsolCountsols)
1668{ /*lint --e{715}*/
1669 SCIP_CONSHDLRDATA* conshdlrdata;
1670
1671 assert(scip != NULL);
1672 assert(conshdlr != NULL);
1673 assert(nconss == 0);
1674 assert(strcmp(SCIPconshdlrGetName(conshdlr), CONSHDLR_NAME) == 0);
1675
1676 conshdlrdata = SCIPconshdlrGetData(conshdlr);
1677 assert(conshdlrdata != NULL );
1678
1679 if( conshdlrdata->active && restart )
1680 {
1681 SCIPerrorMessage("When collecting and counting solutions restarts need to be disabled (presolving/maxrestarts = 0).\n");
1682 SCIPABORT();
1683 return SCIP_INVALIDCALL; /*lint !e527*/
1684 }
1685
1686 return SCIP_OKAY;
1687}
1688
1689/** constraint enforcing method of constraint handler for LP solutions */
1690static
1691SCIP_DECL_CONSENFOLP(consEnfolpCountsols)
1692{ /*lint --e{715}*/
1693 SCIP_CONSHDLRDATA* conshdlrdata;
1694
1695 SCIPdebugMsg(scip, "method SCIP_DECL_CONSENFOLP(consEnfolpCountsols)\n");
1696
1697 assert( scip != NULL );
1698 assert( conshdlr != NULL );
1699 assert( nconss == 0 );
1700
1701 conshdlrdata = SCIPconshdlrGetData(conshdlr);
1702 assert( conshdlrdata != NULL );
1703
1704 if( conshdlrdata->active )
1705 {
1706 if( !solinfeasible )
1707 {
1708 SCIP_SOL* sol;
1709
1710 SCIP_CALL( SCIPcreateLPSol(scip, &sol, NULL ) );
1711
1712 SCIP_CALL( checkSolution(scip, sol, conshdlrdata, result) );
1713 SCIP_CALL( SCIPfreeSol(scip, &sol) );
1714 }
1715 else
1716 *result = SCIP_INFEASIBLE;
1717 }
1718 else
1719 *result = SCIP_FEASIBLE;
1720
1721 assert( !conshdlrdata->active || *result == SCIP_INFEASIBLE || *result == SCIP_CUTOFF );
1722
1723 return SCIP_OKAY;
1724}
1725
1726/** constraint enforcing method of constraint handler for relaxation solutions */
1727static
1728SCIP_DECL_CONSENFORELAX(consEnforelaxCountsols)
1729{ /*lint --e{715}*/
1730 SCIP_CONSHDLRDATA* conshdlrdata;
1731
1732 SCIPdebugMsg(scip, "method SCIP_DECL_CONSENFORELAX(consEnfolpCountsols)\n");
1733
1734 assert( scip != NULL );
1735 assert( conshdlr != NULL );
1736 assert( nconss == 0 );
1737
1738 conshdlrdata = SCIPconshdlrGetData(conshdlr);
1739 assert( conshdlrdata != NULL );
1740
1741 if( conshdlrdata->active )
1742 {
1743 if( !solinfeasible )
1744 {
1745 SCIP_CALL( checkSolution(scip, sol, conshdlrdata, result) );
1746 }
1747 else
1748 *result = SCIP_INFEASIBLE;
1749 }
1750 else
1751 *result = SCIP_FEASIBLE;
1752
1753 assert( !conshdlrdata->active || *result == SCIP_INFEASIBLE || *result == SCIP_CUTOFF );
1754
1755 return SCIP_OKAY;
1756}
1757
1758/** constraint enforcing method of constraint handler for pseudo solutions */
1759static
1760SCIP_DECL_CONSENFOPS(consEnfopsCountsols)
1761{ /*lint --e{715}*/
1762 SCIP_CONSHDLRDATA* conshdlrdata;
1763
1764 SCIPdebugMsg(scip, "method SCIP_DECL_CONSENFOPS(consEnfopsCountsols)\n");
1765
1766 assert( scip != NULL );
1767 assert( conshdlr != NULL );
1768 assert( nconss == 0 );
1769
1770 conshdlrdata = SCIPconshdlrGetData(conshdlr);
1771 assert( conshdlrdata != NULL );
1772
1773 if( conshdlrdata->active )
1774 {
1775 if( !solinfeasible )
1776 {
1777 SCIP_SOL* sol;
1778
1780
1781 SCIP_CALL( checkSolution(scip, sol, conshdlrdata, result) );
1782 SCIP_CALL( SCIPfreeSol(scip, &sol) );
1783 }
1784 else
1785 *result = SCIP_INFEASIBLE;
1786 }
1787 else
1788 *result = SCIP_FEASIBLE;
1789
1790 assert( !conshdlrdata->active || *result == SCIP_INFEASIBLE || *result == SCIP_CUTOFF );
1791
1792 return SCIP_OKAY;
1793}
1794
1795
1796/** feasibility check method of constraint handler for integral solutions */
1797static
1798SCIP_DECL_CONSCHECK(consCheckCountsols)
1799{ /*lint --e{715}*/
1800 /**@todo non-trivial solutions which are only checked have to be ignored since it is unknown how they are generated;
1801 * calculating heuristic solutions should be avoided */
1802 SCIP_CONSHDLRDATA* conshdlrdata;
1803
1804 SCIPdebugMsg(scip, "method SCIP_DECL_CONSCHECK(consCheckCountsols)\n");
1805
1806 conshdlrdata = SCIPconshdlrGetData(conshdlr);
1807 assert( conshdlrdata != NULL );
1808
1809 if( conshdlrdata->active )
1810 {
1811 /* count empty solution */
1812 if( SCIPgetNVars(scip) == 0 )
1813 SCIP_CALL( checkSolution(scip, sol, conshdlrdata, result) );
1814 else if( !conshdlrdata->warning )
1815 {
1816 SCIPwarningMessage(scip, "a non-trivial solution comes in over <SCIP_DECL_CONSCHECK(consCheckCountsols)>; currently these solutions are ignored.\n");
1817 conshdlrdata->warning = TRUE;
1818 }
1819
1820 *result = SCIP_INFEASIBLE;
1821 }
1822 else
1823 *result = SCIP_FEASIBLE;
1824
1825 return SCIP_OKAY;
1826}
1827
1828
1829/** variable rounding lock method of constraint handler */
1830static
1831SCIP_DECL_CONSLOCK(consLockCountsols)
1832{ /*lint --e{715}*/
1833 return SCIP_OKAY;
1834}
1835
1836
1837/*
1838 * Callback methods and local method for dialogs
1839 */
1840
1841/** dialog execution method for the count command */
1842SCIP_DECL_DIALOGEXEC(SCIPdialogExecCountPresolve)
1843{ /*lint --e{715}*/
1845 int usesymmetry;
1846
1847 SCIP_CALL( SCIPgetIntParam(scip, "misc/usesymmetry", &usesymmetry) );
1848
1849 if ( usesymmetry != 0 )
1850 {
1851 int symcomptiming = 2;
1852
1853 /* get timing of symmetry computation */
1854 if ( ((unsigned) usesymmetry & SYM_HANDLETYPE_SYMCONS) != 0 )
1855 {
1856 SCIP_CALL( SCIPgetIntParam(scip, "propagating/symmetry/addconsstiming", &symcomptiming) );
1857 }
1858 else if ( usesymmetry == 2 )
1859 {
1860 SCIP_CALL( SCIPgetIntParam(scip, "propagating/symmetry/ofsymcomptiming", &symcomptiming) );
1861 }
1862
1863 if ( symcomptiming < SYM_TIMING_AFTERPRESOL &&
1865 {
1866 SCIPerrorMessage("Symmetry handling and solution counting are not compatible. " \
1867 "You might want to disable symmetry by setting parameter <misc/usesymmetry> to 0.\n");
1868
1869 return SCIP_INVALIDCALL;
1870 }
1871
1872 SCIPwarningMessage(scip, "Symmetry handling has been deactivated since it is not compatible with counting.\n");
1873 SCIPwarningMessage(scip, "=> counting forces parameter <misc/usesymmetry> to 0.\n");
1874
1875 SCIP_CALL( SCIPsetIntParam(scip, "misc/usesymmetry", 0) );
1876 }
1877
1878 SCIP_CALL( SCIPdialoghdlrAddHistory(dialoghdlr, dialog, NULL, FALSE) );
1879 SCIPdialogMessage(scip, NULL, "\n");
1880 SCIP_CALL( SCIPgetBoolParam(scip, "constraints/" CONSHDLR_NAME "/active", &active) );
1881
1882 switch( SCIPgetStage(scip) )
1883 {
1884 case SCIP_STAGE_INIT:
1885 SCIPdialogMessage(scip, NULL, "no problem exists\n");
1886 break;
1887
1888 case SCIP_STAGE_PROBLEM:
1889 /* activate constraint handler cons_countsols */
1890 if( !active )
1891 {
1892 SCIP_CALL( SCIPsetBoolParam(scip, "constraints/" CONSHDLR_NAME "/active", TRUE) );
1893 }
1894 /*lint -fallthrough*/
1897 /* presolve problem */
1899
1900 /* reset cons_countsols activation */
1901 if( !active )
1902 {
1903 SCIP_CALL( SCIPsetBoolParam(scip, "constraints/" CONSHDLR_NAME "/active", FALSE) );
1904 }
1905 break;
1906
1908 case SCIP_STAGE_SOLVING:
1909 SCIPdialogMessage(scip, NULL, "problem is already presolved\n");
1910 break;
1911
1912 case SCIP_STAGE_SOLVED:
1913 SCIPdialogMessage(scip, NULL, "problem is already (pre)solved\n");
1914 break;
1915
1922 case SCIP_STAGE_FREE:
1923 default:
1924 SCIPerrorMessage("invalid SCIP stage\n");
1925 return SCIP_INVALIDCALL;
1926 } /*lint --e{616}*/
1927
1928 SCIPdialogMessage(scip, NULL, "\n");
1929 *nextdialog = SCIPdialoghdlrGetRoot(dialoghdlr);
1930
1931 return SCIP_OKAY;
1932}
1933
1934/** dialog execution method for the count command */
1935SCIP_DECL_DIALOGEXEC(SCIPdialogExecCount)
1936{ /*lint --e{715}*/
1937 SCIP_RETCODE retcode;
1939
1940 SCIP_Bool valid;
1941 SCIP_Longint nsols;
1942 int displayprimalbound;
1943 int displaygap;
1944 int displaysols;
1945 int displayfeasST;
1946 int nrestarts;
1947 int usesymmetry;
1948
1949 SCIP_CALL( SCIPdialoghdlrAddHistory(dialoghdlr, dialog, NULL, FALSE) );
1950 SCIPdialogMessage(scip, NULL, "\n");
1951 SCIP_CALL( SCIPgetBoolParam(scip, "constraints/" CONSHDLR_NAME "/active", &active) );
1952 SCIP_CALL( SCIPgetIntParam(scip, "presolving/maxrestarts", &nrestarts) );
1953
1954 if( nrestarts != 0 )
1955 {
1956 /* need to disable restarts, since collecting solutions won't work, but also the capturing for variables is not
1957 * correctly handled
1958 */
1959 SCIPwarningMessage(scip, "counting forces parameter <presolving/maxrestarts> to 0.\n");
1960 if( SCIPisParamFixed(scip, "presolving/maxrestarts") )
1961 {
1962 SCIP_CALL( SCIPunfixParam(scip, "presolving/maxrestarts") );
1963 }
1964 SCIP_CALL( SCIPsetIntParam(scip, "presolving/maxrestarts", 0) );
1965 }
1966
1967 SCIP_CALL( SCIPgetIntParam(scip, "misc/usesymmetry", &usesymmetry) );
1968
1969 if ( usesymmetry != 0 )
1970 {
1971 int symcomptiming = 2;
1972
1973 /* get timing of symmetry computation */
1974 if ( ((unsigned) usesymmetry & SYM_HANDLETYPE_SYMCONS) != 0 )
1975 {
1976 SCIP_CALL( SCIPgetIntParam(scip, "propagating/symmetry/addconsstiming", &symcomptiming) );
1977 }
1978 else if ( usesymmetry == 2 )
1979 {
1980 SCIP_CALL( SCIPgetIntParam(scip, "propagating/symmetry/ofsymcomptiming", &symcomptiming) );
1981 }
1982
1983 if ( symcomptiming < SYM_TIMING_AFTERPRESOL &&
1985 {
1986 SCIPerrorMessage("Symmetry handling and solution counting are not compatible. " \
1987 "You might want to disable symmetry by setting parameter <misc/usesymmetry> to 0.\n");
1988
1989 return SCIP_INVALIDCALL;
1990 }
1991
1992 SCIPwarningMessage(scip, "Symmetry handling has been deactivated since it is not compatible with counting.\n");
1993 SCIPwarningMessage(scip, "=> counting forces parameter <misc/usesymmetry> to 0.\n");
1994
1995 SCIP_CALL( SCIPsetIntParam(scip, "misc/usesymmetry", 0) );
1996 }
1997
1998 switch( SCIPgetStage(scip) )
1999 {
2000 case SCIP_STAGE_INIT:
2001 SCIPdialogMessage(scip, NULL, "no problem exists\n");
2002 break;
2003
2004 case SCIP_STAGE_PROBLEM:
2005 /* activate constraint handler cons_countsols */
2006 if( !active )
2007 {
2008 SCIP_CALL( SCIPsetBoolParam(scip, "constraints/" CONSHDLR_NAME "/active", TRUE) );
2009 }
2010 /*lint -fallthrough*/
2013 /* presolve problem */
2015 /*lint -fallthrough*/
2017 /* reset activity status of constraint handler cons_countsols */
2018 if( !active )
2019 {
2020 SCIP_CALL( SCIPsetBoolParam(scip, "constraints/" CONSHDLR_NAME "/active", FALSE) );
2021 }
2022 /*lint -fallthrough*/
2023 case SCIP_STAGE_SOLVING:
2024 /* check if the problem contains continuous variables */
2025 if( SCIPgetNContVars(scip) != 0 )
2026 {
2028 "Problem contains continuous variables (after presolving). Counting projection to integral variables!\n");
2029 }
2030
2031 /* turn off primal bound and gap column */
2032 SCIP_CALL( SCIPgetIntParam(scip, "display/primalbound/active", &displayprimalbound) );
2033 if( displayprimalbound != 0 )
2034 {
2035 SCIP_CALL( SCIPsetIntParam(scip, "display/primalbound/active", 0) );
2036 }
2037 SCIP_CALL( SCIPgetIntParam(scip, "display/gap/active", &displaygap) );
2038 if( displaygap != 0 )
2039 {
2040 SCIP_CALL( SCIPsetIntParam(scip, "display/gap/active", 0) );
2041 }
2042
2043 /* turn on sols and feasST column */
2044 SCIP_CALL( SCIPgetIntParam(scip, "display/sols/active", &displaysols) );
2045 if( displayprimalbound != 2 )
2046 {
2047 SCIP_CALL( SCIPsetIntParam(scip, "display/sols/active", 2) );
2048 }
2049 SCIP_CALL( SCIPgetIntParam(scip, "display/feasST/active", &displayfeasST) );
2050 if( displayprimalbound != 2 )
2051 {
2052 SCIP_CALL( SCIPsetIntParam(scip, "display/feasST/active", 2) );
2053 }
2054
2055 /* find the countsols constraint handler */
2056 assert( SCIPfindConshdlr(scip, CONSHDLR_NAME) != NULL );
2057
2058 retcode = SCIPcount(scip);
2059
2060 valid = FALSE;
2061 nsols = SCIPgetNCountedSols(scip, &valid);
2062
2063 if( valid )
2064 SCIPdialogMessage(scip, NULL, "Feasible Solutions : %" SCIP_LONGINT_FORMAT "", nsols);
2065 else
2066 {
2067 char* buffer;
2068 int buffersize = SCIP_MAXSTRLEN;
2069 int requiredsize;
2070
2071 SCIP_CALL( SCIPallocBufferArray(scip, &buffer, buffersize) );
2072 SCIPgetNCountedSolsstr(scip, &buffer, buffersize, &requiredsize);
2073
2074 if( requiredsize > buffersize )
2075 {
2076 SCIP_CALL( SCIPreallocBufferArray(scip, &buffer, requiredsize) );
2077 SCIPgetNCountedSolsstr(scip, &buffer, buffersize, &requiredsize);
2078 }
2079
2080 assert( buffersize >= requiredsize );
2081 SCIPdialogMessage(scip, NULL, "Feasible Solutions : %s", buffer);
2082
2083 SCIPfreeBufferArray(scip, &buffer);
2084 }
2085
2086 SCIPdialogMessage(scip, NULL, " (%" SCIP_LONGINT_FORMAT " non-trivial feasible subtrees)\n", SCIPgetNCountedFeasSubtrees(scip));
2087
2088 *nextdialog = SCIPdialoghdlrGetRoot(dialoghdlr);
2089
2090 /* reset display columns */
2091 if( displayprimalbound != 0 )
2092 {
2093 SCIP_CALL( SCIPsetIntParam(scip, "display/primalbound/active", displayprimalbound) );
2094 }
2095 if( displaygap != 0 )
2096 {
2097 SCIP_CALL( SCIPsetIntParam(scip, "display/gap/active", displaygap) );
2098 }
2099
2100 /* reset sols and feasST column */
2101 if( displaysols != 2 )
2102 {
2103 SCIP_CALL( SCIPsetIntParam(scip, "display/sols/active", displaysols) );
2104 }
2105 if( displayfeasST != 2 )
2106 {
2107 SCIP_CALL( SCIPsetIntParam(scip, "display/feasST/active", displayfeasST) );
2108 }
2109
2110 /* reset cons_countsols activation */
2111 if( !active )
2112 {
2113 SCIP_CALL( SCIPsetBoolParam(scip, "constraints/" CONSHDLR_NAME "/active", FALSE) );
2114 }
2115
2116 /* evaluate retcode */
2117 SCIP_CALL( retcode );
2118 break;
2119
2120 case SCIP_STAGE_SOLVED:
2121 SCIPdialogMessage(scip, NULL, "problem is already solved\n");
2122 break;
2123
2130 case SCIP_STAGE_FREE:
2131 default:
2132 SCIPerrorMessage("invalid SCIP stage\n");
2133 return SCIP_INVALIDCALL;
2134 } /*lint --e{616}*/
2135
2136 SCIPdialogMessage(scip, NULL, "\n");
2137 *nextdialog = SCIPdialoghdlrGetRoot(dialoghdlr);
2138
2139 return SCIP_OKAY;
2140}
2141
2142/** comparison method for sorting variables by non-decreasing w.r.t. problem index */
2143static
2144SCIP_DECL_SORTPTRCOMP(varCompProbindex)
2145{
2146 SCIP_VAR* var1;
2147 SCIP_VAR* var2;
2148
2149 var1 = (SCIP_VAR*)elem1;
2150 var2 = (SCIP_VAR*)elem2;
2151
2152 assert(var1 != NULL);
2153 assert(var2 != NULL);
2154
2155 if( SCIPvarGetProbindex(var1) < SCIPvarGetProbindex(var2) )
2156 return -1;
2157 else if( SCIPvarGetProbindex(var1) > SCIPvarGetProbindex(var2) )
2158 return +1;
2159 else
2160 {
2161 assert(var1 == var2 || (SCIPvarGetProbindex(var1) == -1 && SCIPvarGetProbindex(var2) == -1));
2162 return 0;
2163 }
2164}
2165
2166/** expands the sparse solutions and writes them to the file */
2167static
2169 SCIP* scip, /**< SCIP data structure */
2170 FILE* file, /**< file handler */
2171 SCIP_VAR** allvars, /**< SCIP variables */
2172 int nallvars, /**< number of all variables */
2173 SCIP_VAR** activevars, /**< SCIP variables */
2174 int nactivevars, /**< number of active variables */
2175 SCIP_HASHMAP* hashmap, /**< hashmap from active solution variable to the position in the active
2176 * variables array
2177 */
2178 SCIP_SPARSESOL** sols, /**< sparse solutions to expands and write */
2179 int nsols /**< number of sparse solutions */
2180 )
2181{
2182 SCIP_SPARSESOL* sparsesol;
2183 SCIP_VAR** vars;
2185 SCIP_Longint* sol;
2186 SCIP_Longint solcnt;
2187 int s;
2188 int v;
2189
2190 assert(scip != NULL);
2191 assert(file != NULL);
2192 assert(hashmap != NULL);
2193 assert(allvars != NULL || nallvars == 0);
2194 assert(activevars != NULL || nactivevars == 0);
2195 assert(sols != NULL || nsols == 0);
2196
2197 solcnt = 0;
2198
2199 /* get memory to store active solution */
2200 SCIP_CALL( SCIPallocBufferArray(scip, &sol, nactivevars) );
2201 SCIP_CALL( SCIPallocBufferArray(scip, &vars, nactivevars) );
2202 SCIP_CALL( SCIPallocBufferArray(scip, &scalars, nactivevars) );
2203
2204 /* loop over all sparse solutions */
2205 for( s = 0; s < nsols; ++s )
2206 {
2207 sparsesol = sols[s]; /*lint !e613*/
2208 assert(sparsesol != NULL);
2209 assert(SCIPsparseSolGetNVars(sparsesol) == nactivevars);
2210
2211 /* get first solution of the sparse solution */
2212 SCIPsparseSolGetFirstSol(sparsesol, sol, nactivevars);
2213
2214 do
2215 {
2216 SCIP_Real objval;
2217
2218 solcnt++;
2219
2220 /* print solution number */
2221 SCIPinfoMessage(scip, file, "%d(%" SCIP_LONGINT_FORMAT "), ", s+1, solcnt);
2222
2223 objval = 0.0;
2224
2225 /* write none active variables */
2226 for( v = 0; v < nallvars; ++v )
2227 {
2228 SCIP_Real constant;
2229 SCIP_Real realvalue;
2230 int requiredsize;
2231 int nvars;
2232 int idx;
2233 int i;
2234
2235 vars[0] = allvars[v]; /*lint !e613*/
2236 scalars[0] = 1.0;
2237 nvars = 1;
2238 constant = 0.0;
2239
2240 SCIP_CALL( SCIPgetProbvarLinearSum(scip, vars, scalars, &nvars, nallvars, &constant, &requiredsize, TRUE) );
2241 assert(requiredsize <= nallvars);
2242 assert(nvars <= nactivevars);
2243
2244 realvalue = constant;
2245
2246 for( i = 0; i < nvars; ++i )
2247 {
2248 assert(SCIPhashmapExists(hashmap, vars[i]));
2249 idx = SCIPhashmapGetImageInt(hashmap, vars[i]) - 1;
2250 assert(0 <= idx && idx < nactivevars);
2251 assert(activevars[idx] == vars[i]); /*lint !e613*/
2252
2253 objval += SCIPvarGetObj(vars[i]) * sol[idx];
2254 realvalue += scalars[i] * sol[idx];
2255 }
2256 assert(SCIPisIntegral(scip, realvalue));
2257
2258 SCIPinfoMessage(scip, file, "%g, ", realvalue);
2259 }
2260
2261 /* transform objective value into original problem space */
2262 objval = SCIPretransformObj(scip, objval);
2263
2264 /* output the objective value of the solution */
2265 SCIPinfoMessage(scip, file, "%g\n", objval);
2266 }
2267 while( SCIPsparseSolGetNextSol(sparsesol, sol, nactivevars) );
2268 }
2269
2270 /* free buffer arrays */
2272 SCIPfreeBufferArray(scip, &vars);
2274
2275 return SCIP_OKAY;
2276}
2277
2278/** execution method of dialog for writing all solutions */
2279SCIP_DECL_DIALOGEXEC(SCIPdialogExecWriteAllsolutions)
2280{ /*lint --e{715}*/
2281 FILE* file;
2282 SCIP_Longint nsols;
2283 char* filename;
2284 char* word;
2285 SCIP_Bool endoffile;
2286 SCIP_Bool valid;
2287
2288 assert( scip != NULL );
2289
2290 SCIP_CALL( SCIPdialoghdlrAddHistory(dialoghdlr, dialog, NULL, FALSE) );
2291
2292 switch( SCIPgetStage(scip) )
2293 {
2294 case SCIP_STAGE_INIT:
2295 SCIPdialogMessage(scip, NULL, "no problem available\n");
2296 break;
2297 case SCIP_STAGE_PROBLEM:
2300 SCIPdialogMessage(scip, NULL, "the counting process was not started yet\n");
2301 break;
2308 case SCIP_STAGE_SOLVING:
2309 case SCIP_STAGE_SOLVED:
2311 {
2312 SCIP_CONSHDLR* conshdlr;
2313 SCIP_CONSHDLRDATA* conshdlrdata;
2314 int nsparsesols;
2315
2316 valid = FALSE;
2317 nsols = SCIPgetNCountedSols(scip, &valid);
2318
2319 /* find the countsols constraint handler */
2320 conshdlr = SCIPfindConshdlr(scip, CONSHDLR_NAME);
2321 assert( conshdlr != NULL );
2322
2323 conshdlrdata = SCIPconshdlrGetData(conshdlr);
2324 assert( conshdlrdata != NULL );
2325
2326 nsparsesols = conshdlrdata->nsolutions;
2327
2328 if( !valid )
2329 {
2330 /* too many solutions, output not "possible" */
2331 char* buffer;
2332 int buffersize;
2333 int requiredsize;
2334
2335 buffersize = SCIP_MAXSTRLEN;
2336
2337 SCIP_CALL( SCIPallocBufferArray(scip, &buffer, buffersize) );
2338 SCIPgetNCountedSolsstr(scip, &buffer, buffersize, &requiredsize);
2339
2340 if( requiredsize > buffersize )
2341 {
2342 buffersize = requiredsize;
2343 SCIP_CALL( SCIPreallocBufferArray(scip, &buffer, requiredsize) );
2344 SCIPgetNCountedSolsstr(scip, &buffer, buffersize, &requiredsize);
2345 }
2346
2347 assert( buffersize >= requiredsize );
2348 SCIPdialogMessage(scip, NULL, "no output, because of too many feasible solutions : %s\n", buffer);
2349
2350 SCIPfreeBufferArray(scip, &buffer);
2351 }
2352 else if( nsols == 0 )
2353 {
2354 SCIPdialogMessage(scip, NULL, "there are no counted solutions\n");
2355 }
2356 else if( nsparsesols == 0 )
2357 {
2358 SCIPdialogMessage(scip, NULL, "there is no solution collect (set parameter <constraints/countsols/collect> to TRUE)\n");
2359 }
2360 else
2361 {
2362 SCIP_CALL( SCIPdialoghdlrGetWord(dialoghdlr, dialog, "enter filename: ", &word, &endoffile) );
2363
2364 /* copy the filename for later use */
2365 SCIP_CALL( SCIPduplicateBufferArray(scip, &filename, word, (int)strlen(word)+1) );
2366
2367 if( endoffile )
2368 {
2369 *nextdialog = NULL;
2370 return SCIP_OKAY;
2371 }
2372
2373 SCIP_CALL( SCIPdialoghdlrAddHistory(dialoghdlr, dialog, filename, TRUE) );
2374
2375 if( filename[0] != '\0' )
2376 {
2377 file = fopen(filename, "w");
2378
2379 if( file == NULL )
2380 {
2381 SCIPdialogMessage(scip, NULL, "error creating file <%s>\n", filename);
2382 SCIPdialoghdlrClearBuffer(dialoghdlr);
2383 }
2384 else
2385 {
2386 SCIP_SPARSESOL** sparsesols;
2387 SCIP_VAR** origvars;
2388 SCIP_VAR** allvars;
2389 int norigvars;
2390 int nvars;
2391 int v;
2392
2393 SCIP_RETCODE retcode;
2394
2395 /* get sparse solutions defined over the active variables */
2396 nvars = conshdlrdata->nvars;
2397 sparsesols = conshdlrdata->solutions;
2398
2399 /* get original problem variables */
2400 retcode = SCIPallocBufferArray(scip, &origvars, SCIPgetNOrigVars(scip));
2401 if( retcode != SCIP_OKAY )
2402 {
2403 fclose(file);
2404 SCIP_CALL( retcode );
2405 }
2406
2407 norigvars = 0;
2408
2409 for( v = 0; v < SCIPgetNOrigVars(scip); ++v )
2410 {
2412 {
2413 origvars[norigvars] = SCIPgetOrigVars(scip)[v];
2414 norigvars++;
2415 }
2416 }
2417 assert(norigvars == conshdlrdata->nallvars);
2418
2419 retcode = SCIPduplicateBufferArray(scip, &allvars, conshdlrdata->allvars, norigvars);
2420 if( retcode != SCIP_OKAY )
2421 {
2422 fclose(file); /*lint !e449*/
2423 SCIP_CALL( retcode );
2424 }
2425
2426 /* sort original variables array and the corresponding transformed variables w.r.t. the problem index */
2427 SCIPsortDownPtrPtr((void**)allvars, (void**)origvars, varCompProbindex, norigvars);
2428
2429 SCIPdialogMessage(scip, NULL, "saving %" SCIP_LONGINT_FORMAT " (%d) feasible solutions\n", nsols, nsparsesols);
2430
2431 /* first row: output the names of the variables in the given ordering */
2432 SCIPinfoMessage(scip, file, "#, ");
2433
2434 for( v = 0; v < norigvars; ++v )
2435 {
2436#ifndef NDEBUG
2437 {
2438 /* check if the original variable fits to the transformed variable the constraint handler has stored */
2439 SCIP_VAR* transvar;
2440 SCIP_CALL( SCIPgetTransformedVar(scip, origvars[v], &transvar) );
2441 assert(transvar != NULL);
2442 assert(transvar == allvars[v]);
2443 }
2444#endif
2445 SCIPinfoMessage(scip, file, "%s, ", SCIPvarGetName(origvars[v]));
2446 }
2447
2448 SCIPinfoMessage(scip, file, "objval\n");
2449
2450 /* expand and write solution */
2451 retcode = writeExpandedSolutions(scip, file, allvars, conshdlrdata->nallvars, conshdlrdata->vars, nvars, conshdlrdata->hashmap, sparsesols, nsparsesols);
2452 if( retcode != SCIP_OKAY )
2453 {
2454 fclose(file);
2455 SCIP_CALL( retcode );
2456 }
2457 SCIPdialogMessage(scip, NULL, "written solutions information to file <%s>\n", filename);
2458
2459 SCIPfreeBufferArray(scip, &allvars);
2460 SCIPfreeBufferArray(scip, &origvars);
2461
2462 fclose(file);
2463 }
2464
2465 /* free buffer array */
2466 SCIPfreeBufferArray(scip, &filename);
2467 }
2468 }
2469 break;
2470 }
2471 case SCIP_STAGE_FREE:
2472 SCIPerrorMessage("invalid call during SCIP_STAGE_FREE\n");
2473 return SCIP_ERROR;
2474 }
2475
2476 *nextdialog = SCIPdialoghdlrGetRoot(dialoghdlr);
2477
2478 return SCIP_OKAY;
2479}
2480
2481/** create the interactive shell dialogs for the counting process */
2482static
2484 SCIP* scip /**< SCIP data structure */
2485 )
2486{
2487 SCIP_DIALOG* root;
2488 SCIP_DIALOG* dialog;
2489 SCIP_DIALOG* submenu;
2490
2491 root = SCIPgetRootDialog(scip);
2492
2493 /* skip dialogs if they seem to be disabled */
2494 if( root == NULL )
2495 return SCIP_OKAY;
2496
2497 /* add dialog entry for counting */
2498 if( !SCIPdialogHasEntry(root, "count") )
2499 {
2500 SCIP_CALL( SCIPincludeDialog(scip, &dialog, NULL, SCIPdialogExecCount, NULL, NULL,
2501 "count", "count number of feasible solutions", FALSE, NULL) );
2502 SCIP_CALL( SCIPaddDialogEntry(scip, root, dialog) );
2503 SCIP_CALL( SCIPreleaseDialog(scip, &dialog) );
2504 }
2505
2506 /* add dialog entry for counting */
2507 if( !SCIPdialogHasEntry(root, "countpresolve") )
2508 {
2509 SCIP_CALL( SCIPincludeDialog(scip, &dialog, NULL, SCIPdialogExecCountPresolve, NULL, NULL,
2510 "countpresolve", "presolve instance before counting number of feasible solutions", FALSE, NULL) );
2511 SCIP_CALL( SCIPaddDialogEntry(scip, root, dialog) );
2512 SCIP_CALL( SCIPreleaseDialog(scip, &dialog) );
2513 }
2514
2515 /* search for the "write" sub menu to add "allsolutions" dialog */
2516 if( SCIPdialogFindEntry(root, "write", &submenu) != 1 )
2517 {
2518 SCIPerrorMessage("write sub menu not found\n");
2519 return SCIP_PLUGINNOTFOUND;
2520 }
2521 assert(submenu != NULL);
2522
2523 /* add dialog "allsolutions" to sub menu "write" */
2524 if( !SCIPdialogHasEntry(submenu, "allsolutions") )
2525 {
2526 SCIP_CALL( SCIPincludeDialog(scip, &dialog, NULL, SCIPdialogExecWriteAllsolutions, NULL, NULL,
2527 "allsolutions", "write all counted primal solutions to file", FALSE, NULL) );
2528 SCIP_CALL( SCIPaddDialogEntry(scip, submenu, dialog) );
2529 SCIP_CALL( SCIPreleaseDialog(scip, &dialog) );
2530 }
2531
2532 return SCIP_OKAY;
2533}
2534
2535/*
2536 * Callback methods for columns
2537 */
2538
2539/** output method of display column to output file stream 'file' */
2540static
2542{ /*lint --e{715}*/
2543#ifndef NDEBUG
2544 SCIP_CONSHDLR* conshdlr;
2545#endif
2546 SCIP_Longint sols;
2547 SCIP_Bool valid;
2548
2549 assert(disp != NULL);
2550 assert(strcmp(SCIPdispGetName(disp), DISP_SOLS_NAME) == 0);
2551 assert(scip != NULL);
2552
2553#ifndef NDEBUG
2554 conshdlr = SCIPfindConshdlr(scip, CONSHDLR_NAME);
2555 assert( conshdlr != NULL );
2556 assert( SCIPconshdlrGetNConss(conshdlr) == 0 );
2557#endif
2558
2559 sols = SCIPgetNCountedSols(scip, &valid);
2560
2561 if( !valid )
2562 {
2563 SCIPinfoMessage(scip, file, "TooMany");
2564 }
2565 else
2566 {
2568 }
2569
2570 return SCIP_OKAY;
2571}
2572
2573
2574/** output method of display column to output file stream 'file' */
2575static
2576SCIP_DECL_DISPOUTPUT(dispOutputFeasSubtrees)
2577{ /*lint --e{715}*/
2578#ifndef NDEBUG
2579 SCIP_CONSHDLR* conshdlr;
2580#endif
2581
2582 assert(disp != NULL);
2583 assert(scip != NULL);
2584 assert(strcmp(SCIPdispGetName(disp), DISP_CUTS_NAME) == 0);
2585
2586#ifndef NDEBUG
2587 conshdlr = SCIPfindConshdlr(scip, CONSHDLR_NAME);
2588 assert( conshdlr != NULL );
2589 assert( SCIPconshdlrGetNConss(conshdlr) == 0 );
2590#endif
2591
2593
2594 return SCIP_OKAY;
2595}
2596
2597
2598/*
2599 * Interface methods of constraint handler
2600 */
2601
2602/** creates the handler for countsols constraints and includes it in SCIP */
2603static
2605 SCIP* scip, /**< SCIP data structure */
2606 SCIP_Bool dialogs /**< sould count dialogs be added */
2607 )
2608{
2609 /* create countsol constraint handler data */
2610 SCIP_CONSHDLRDATA* conshdlrdata;
2611 SCIP_CONSHDLR* conshdlr;
2612
2613#ifdef SCIP_WITH_GMP
2614 char gmpversion[20];
2615#endif
2616
2617 /* create constraint handler specific data here */
2618 SCIP_CALL( conshdlrdataCreate(scip, &conshdlrdata) );
2619
2620 /* include constraint handler */
2623 consEnfolpCountsols, consEnfopsCountsols, consCheckCountsols, consLockCountsols,
2624 conshdlrdata) );
2625
2626 assert(conshdlr != NULL);
2627
2628 /* set non-fundamental callbacks via specific setter functions */
2629 SCIP_CALL( SCIPsetConshdlrCopy(scip, conshdlr, conshdlrCopyCountsols, consCopyCountsols) );
2630 SCIP_CALL( SCIPsetConshdlrExit(scip, conshdlr, consExitCountsols) );
2631 SCIP_CALL( SCIPsetConshdlrExitsol(scip, conshdlr, consExitsolCountsols) );
2632 SCIP_CALL( SCIPsetConshdlrFree(scip, conshdlr, consFreeCountsols) );
2633 SCIP_CALL( SCIPsetConshdlrInit(scip, conshdlr, consInitCountsols) );
2634 SCIP_CALL( SCIPsetConshdlrInitsol(scip, conshdlr, consInitsolCountsols) );
2635 SCIP_CALL( SCIPsetConshdlrEnforelax(scip, conshdlr, consEnforelaxCountsols) );
2636
2637 /* add countsols constraint handler parameters */
2639 "constraints/" CONSHDLR_NAME "/active",
2640 "is the constraint handler active?",
2641 &conshdlrdata->active, FALSE, DEFAULT_ACTIVE, NULL, NULL));
2643 "constraints/" CONSHDLR_NAME "/sparsetest",
2644 "should the sparse solution test be turned on?",
2645 &conshdlrdata->sparsetest, FALSE, DEFAULT_SPARSETEST, NULL, NULL));
2647 "constraints/" CONSHDLR_NAME "/discardsols",
2648 "is it allowed to discard solutions?",
2649 &conshdlrdata->discardsols, FALSE, DEFAULT_DISCARDSOLS, NULL, NULL));
2651 "constraints/" CONSHDLR_NAME "/collect",
2652 "should the solutions be collected?",
2653 &conshdlrdata->collect, FALSE, DEFAULT_COLLECT, NULL, NULL));
2655 "constraints/" CONSHDLR_NAME "/sollimit",
2656 "counting stops, if the given number of solutions were found (-1: no limit)",
2657 &conshdlrdata->sollimit, FALSE, DEFAULT_SOLLIMIT, -1LL, SCIP_LONGINT_MAX, NULL, NULL));
2658
2659 /* create the interactive shell dialogs for the counting process */
2660 if( dialogs )
2661 {
2663 }
2664
2665 /* include display column */
2667 NULL, NULL, NULL, NULL, NULL, NULL, dispOutputSols,
2670 NULL, NULL, NULL, NULL, NULL, NULL, dispOutputFeasSubtrees,
2672
2673#ifdef SCIP_WITH_GMP
2674#ifdef mpir_version
2675 /* add info about using MPIR to external codes information */
2676 (void) SCIPsnprintf(gmpversion, (int) sizeof(gmpversion), "MPIR %s", mpir_version);
2677 SCIP_CALL( SCIPincludeExternalCodeInformation(scip, gmpversion, "Multiple Precision Integers and Rationals Library developed by W. Hart (mpir.org)") );
2678#else
2679 /* add info about using GMP to external codes information */
2680 (void) SCIPsnprintf(gmpversion, (int) sizeof(gmpversion), "GMP %s", gmp_version);
2681 SCIP_CALL( SCIPincludeExternalCodeInformation(scip, gmpversion, "GNU Multiple Precision Arithmetic Library developed by T. Granlund (gmplib.org)") );
2682#endif
2683#endif
2684
2685 return SCIP_OKAY;
2686}
2687
2688/** creates the handler for countsols constraints and includes it in SCIP */
2690 SCIP* scip /**< SCIP data structure */
2691 )
2692{
2693 /* include constraint handler including the count dialog */
2695
2696 return SCIP_OKAY;
2697}
2698
2699
2700/** execute counting */
2702 SCIP* scip /**< SCIP data structure */
2703 )
2704{
2706
2707 /* activate constraint handler cons_countsols */
2708 SCIP_CALL( SCIPgetBoolParam(scip, "constraints/" CONSHDLR_NAME "/active", &active) );
2709 if( !active )
2710 {
2711 SCIP_CALL( SCIPsetBoolParam(scip, "constraints/" CONSHDLR_NAME "/active", TRUE) );
2712 }
2713
2714 /* check if the parameter setting allows a valid counting process */
2716
2717 /* start the solving process */
2719
2720 /* reset activity status of constraint handler cons_countsols */
2721 if( !active )
2722 {
2723 SCIP_CALL( SCIPsetBoolParam(scip, "constraints/" CONSHDLR_NAME "/active", FALSE) );
2724 }
2725
2726 return SCIP_OKAY;
2727}
2728
2729
2730/** returns number of feasible solutions found as SCIP_Longint; if the number does not fit into
2731 * a SCIP_Longint the valid flag is set to FALSE
2732 */
2734 SCIP* scip, /**< SCIP data structure */
2735 SCIP_Bool* valid /**< pointer to store if the return value is valid */
2736 )
2737{
2738 SCIP_CONSHDLR* conshdlr;
2739 SCIP_CONSHDLRDATA* conshdlrdata;
2740
2741 /* find the countsols constraint handler */
2742 conshdlr = SCIPfindConshdlr(scip, CONSHDLR_NAME);
2743 assert( conshdlr != NULL );
2744
2745 conshdlrdata = SCIPconshdlrGetData(conshdlr);
2746 assert( conshdlrdata != NULL );
2747
2748 return getNCountedSols(conshdlrdata->nsols, valid);
2749}
2750
2751
2752/** puts the number of counted solutions in the given char* buffer */
2754 SCIP* scip, /**< SCIP data structure */
2755 char** buffer, /**< buffer to store the number for counted solutions */
2756 int buffersize, /**< buffer size */
2757 int* requiredsize /**< pointer to store the required size */
2758 )
2759{
2760 SCIP_CONSHDLR* conshdlr;
2761 SCIP_CONSHDLRDATA* conshdlrdata;
2762
2763 /* find the countsols constraint handler */
2764 conshdlr = SCIPfindConshdlr(scip, CONSHDLR_NAME);
2765 assert( conshdlr != NULL );
2766
2767 conshdlrdata = SCIPconshdlrGetData(conshdlr);
2768 assert( conshdlrdata != NULL );
2769
2770#ifdef SCIP_WITH_GMP
2771 /* size must be by two larger than the length of the string, since there need to be storage for a sign and a
2772 * null-termination
2773 */
2774 assert(0 <= (int) (mpz_sizeinbase( conshdlrdata->nsols, 10 ) + 2));
2775 *requiredsize = (int) (mpz_sizeinbase( conshdlrdata->nsols, 10 ) + 2);
2776 if( *requiredsize <= buffersize)
2777 toString(conshdlrdata->nsols, buffer, buffersize);
2778#else
2779 if( conshdlrdata->nsols < pow(10.0, (double)buffersize) )
2780 {
2781 toString(conshdlrdata->nsols, buffer, buffersize);
2782 *requiredsize = (int)strlen(*buffer);
2783 }
2784 else
2785 *requiredsize = 21;
2786#endif
2787}
2788
2789
2790/** returns number of counted non trivial feasible subtrees */
2792 SCIP* scip /**< SCIP data structure */
2793 )
2794{
2795 SCIP_CONSHDLR* conshdlr;
2796 SCIP_CONSHDLRDATA* conshdlrdata;
2797
2798 assert( scip != NULL );
2799
2800 /* find the countsols constraint handler */
2801 conshdlr = SCIPfindConshdlr(scip, CONSHDLR_NAME);
2802 assert( conshdlr != NULL );
2803
2804 conshdlrdata = SCIPconshdlrGetData(conshdlr);
2805 assert( conshdlrdata != NULL );
2806
2807 return conshdlrdata->feasST;
2808}
2809
2810
2811/** Method to get the sparse solution.
2812 *
2813 * @note You get the pointer to the sparse solutions stored in the constraint handler (not a copy).
2814 *
2815 * @note The sparse solutions are stored w.r.t. the active variables. There are the variables which have not been removed
2816 * during presolving. For none active variables the value has to be computed depending on their aggregation
2817 * type. See for more details about that \ref COLLECTALLFEASEBLES.
2818 */
2820 SCIP* scip, /**< SCIP data structure */
2821 SCIP_VAR*** vars, /**< pointer to active variable array defining to variable order */
2822 int* nvars, /**< number of active variables */
2823 SCIP_SPARSESOL*** sols, /**< pointer to the solutions */
2824 int* nsols /**< pointer to number of solutions */
2825 )
2826{
2827 SCIP_CONSHDLR* conshdlr;
2828 SCIP_CONSHDLRDATA* conshdlrdata;
2829
2830 assert( scip != NULL );
2831
2832 /* find the countsols constraint handler */
2833 conshdlr = SCIPfindConshdlr(scip, CONSHDLR_NAME);
2834 assert( conshdlr != NULL );
2835
2836 conshdlrdata = SCIPconshdlrGetData(conshdlr);
2837 assert( conshdlrdata != NULL );
2838
2839 *vars = conshdlrdata->vars;
2840 *nvars = conshdlrdata->nvars;
2841 *sols = conshdlrdata->solutions;
2842 *nsols = conshdlrdata->nsolutions;
2843}
2844
2845/** setting SCIP parameters for such that a valid counting process is possible */
2847 SCIP* scip /**< SCIP data structure */
2848 )
2849{
2851 return SCIP_OKAY;
2852}
static GRAPHNODE ** active
SCIP_VAR * h
Definition: circlepacking.c:68
constraint handler for bound disjunction constraints
#define DISP_CUTS_NAME
#define DISP_SOLS_STRIPLINE
static SCIP_DECL_CONSEXITSOL(consExitsolCountsols)
static SCIP_Longint getNCountedSols(Int value, SCIP_Bool *valid)
static SCIP_DECL_SORTPTRCOMP(varCompProbindex)
#define CONSHDLR_NEEDSCONS
static SCIP_RETCODE createCountDialog(SCIP *scip)
static SCIP_DECL_DISPOUTPUT(dispOutputSols)
#define DEFAULT_DISCARDSOLS
#define CONSHDLR_CHECKPRIORITY
static void toString(Int value, char **buffer, int buffersize)
#define DISP_SOLS_PRIORITY
#define CONSHDLR_DESC
static SCIP_RETCODE checkFeasSubtree(SCIP *scip, SCIP_SOL *sol, SCIP_Bool *feasible)
#define DISP_CUTS_POSITION
static SCIP_DECL_CONSINITSOL(consInitsolCountsols)
static SCIP_RETCODE writeExpandedSolutions(SCIP *scip, FILE *file, SCIP_VAR **allvars, int nallvars, SCIP_VAR **activevars, int nactivevars, SCIP_HASHMAP *hashmap, SCIP_SPARSESOL **sols, int nsols)
static SCIP_RETCODE checkLogicor(SCIP *scip, SCIP_CONSHDLR *conshdlr, int nconss, SCIP_Bool *satisfied)
static SCIP_DECL_CONSENFOPS(consEnfopsCountsols)
static SCIP_DECL_CONSFREE(consFreeCountsols)
static void multInt(Int *value, SCIP_Longint factor)
#define DISP_CUTS_WIDTH
static SCIP_DECL_CONSEXIT(consExitCountsols)
#define DISP_CUTS_HEADER
static void addInt(Int *value, Int *summand)
static SCIP_RETCODE checkSolution(SCIP *scip, SCIP_SOL *sol, SCIP_CONSHDLRDATA *conshdlrdata, SCIP_RESULT *result)
#define CUTOFF_CONSTRAINT(x)
#define DISP_SOLS_WIDTH
#define DISP_SOLS_NAME
static SCIP_RETCODE collectSolution(SCIP *scip, SCIP_CONSHDLRDATA *conshdlrdata, SCIP_SOL *sol)
#define DEFAULT_SPARSETEST
static void setInt(Int *value, SCIP_Longint newvalue)
static SCIP_RETCODE checkParameters(SCIP *scip)
#define DISP_SOLS_HEADER
static SCIP_DECL_CONSLOCK(consLockCountsols)
static void checkSolutionOrig(SCIP *scip, SCIP_SOL *sol, SCIP_CONSHDLRDATA *conshdlrdata)
static SCIP_DECL_CONSHDLRCOPY(conshdlrCopyCountsols)
static void allocInt(Int *value)
#define DEFAULT_COLLECT
#define DISP_CUTS_DESC
#define DISP_CUTS_PRIORITY
static void addOne(Int *value)
static void freeInt(Int *value)
static SCIP_RETCODE checkVarbound(SCIP *scip, SCIP_CONSHDLR *conshdlr, int nconss, SCIP_Bool *satisfied)
#define DISP_CUTS_STRIPLINE
static SCIP_DECL_CONSENFORELAX(consEnforelaxCountsols)
static SCIP_RETCODE conshdlrdataCreate(SCIP *scip, SCIP_CONSHDLRDATA **conshdlrdata)
static void setPowerOfTwo(Int *value, SCIP_Longint exponent)
static SCIP_RETCODE includeConshdlrCountsols(SCIP *scip, SCIP_Bool dialogs)
#define DEFAULT_SOLLIMIT
#define CONSHDLR_EAGERFREQ
static SCIP_Bool varIsUnfixedLocal(SCIP_VAR *var)
static SCIP_DECL_CONSINIT(consInitCountsols)
#define DISP_SOLS_DESC
static SCIP_DECL_CONSCHECK(consCheckCountsols)
static SCIP_DECL_CONSENFOLP(consEnfolpCountsols)
static SCIP_RETCODE checkBounddisjunction(SCIP *scip, SCIP_CONSHDLR *conshdlr, int nconss, SCIP_Bool *satisfied)
#define CONSHDLR_ENFOPRIORITY
SCIP_Longint Int
#define CONSHDLR_NAME
static SCIP_RETCODE checkKnapsack(SCIP *scip, SCIP_CONSHDLR *conshdlr, int nconss, SCIP_Bool *satisfied)
#define consCopyCountsols
#define DEFAULT_ACTIVE
static SCIP_RETCODE countSparseSol(SCIP *scip, SCIP_SOL *sol, SCIP_Bool feasible, SCIP_CONSHDLRDATA *conshdlrdata, SCIP_RESULT *result)
#define DISP_SOLS_POSITION
Constraint handler for counting feasible solutions.
Constraint handler for knapsack constraints of the form , x binary and .
Constraint handler for logicor constraints (equivalent to set covering, but algorithms are suited fo...
Constraint handler for the set partitioning / packing / covering constraints .
Constraint handler for variable bound constraints .
#define NULL
Definition: def.h:266
#define SCIP_MAXSTRLEN
Definition: def.h:287
#define SCIP_Longint
Definition: def.h:157
#define SCIP_Bool
Definition: def.h:91
#define SCIP_Real
Definition: def.h:172
#define TRUE
Definition: def.h:93
#define FALSE
Definition: def.h:94
#define SCIP_LONGINT_FORMAT
Definition: def.h:164
#define SCIPABORT()
Definition: def.h:345
#define SCIP_LONGINT_MAX
Definition: def.h:158
#define SCIP_CALL(x)
Definition: def.h:373
#define SCIP_CALL_FINALLY(x, y)
Definition: def.h:415
default user interface dialog
int SCIPgetNVarsKnapsack(SCIP *scip, SCIP_CONS *cons)
SCIP_Real SCIPgetVbdcoefVarbound(SCIP *scip, SCIP_CONS *cons)
int SCIPgetNVarsLogicor(SCIP *scip, SCIP_CONS *cons)
SCIP_Real * SCIPgetBoundsBounddisjunction(SCIP *scip, SCIP_CONS *cons)
void SCIPgetNCountedSolsstr(SCIP *scip, char **buffer, int buffersize, int *requiredsize)
SCIP_RETCODE SCIPcreateConsBounddisjunction(SCIP *scip, SCIP_CONS **cons, const char *name, int nvars, SCIP_VAR **vars, SCIP_BOUNDTYPE *boundtypes, SCIP_Real *bounds, SCIP_Bool initial, SCIP_Bool separate, SCIP_Bool enforce, SCIP_Bool check, SCIP_Bool propagate, SCIP_Bool local, SCIP_Bool modifiable, SCIP_Bool dynamic, SCIP_Bool removable, SCIP_Bool stickingatnode)
int SCIPgetNVarsBounddisjunction(SCIP *scip, SCIP_CONS *cons)
SCIP_VAR * SCIPgetVbdvarVarbound(SCIP *scip, SCIP_CONS *cons)
SCIP_BOUNDTYPE * SCIPgetBoundtypesBounddisjunction(SCIP *scip, SCIP_CONS *cons)
SCIP_VAR * SCIPgetVarVarbound(SCIP *scip, SCIP_CONS *cons)
SCIP_Longint SCIPgetNCountedSols(SCIP *scip, SCIP_Bool *valid)
SCIP_Longint * SCIPgetWeightsKnapsack(SCIP *scip, SCIP_CONS *cons)
SCIP_Longint SCIPgetCapacityKnapsack(SCIP *scip, SCIP_CONS *cons)
SCIP_Real SCIPgetLhsVarbound(SCIP *scip, SCIP_CONS *cons)
SCIP_Longint SCIPgetNCountedFeasSubtrees(SCIP *scip)
SCIP_VAR ** SCIPgetVarsLogicor(SCIP *scip, SCIP_CONS *cons)
SCIP_Real SCIPgetRhsVarbound(SCIP *scip, SCIP_CONS *cons)
SCIP_VAR ** SCIPgetVarsKnapsack(SCIP *scip, SCIP_CONS *cons)
SCIP_RETCODE SCIPcreateConsSetcover(SCIP *scip, SCIP_CONS **cons, const char *name, int nvars, SCIP_VAR **vars, SCIP_Bool initial, SCIP_Bool separate, SCIP_Bool enforce, SCIP_Bool check, SCIP_Bool propagate, SCIP_Bool local, SCIP_Bool modifiable, SCIP_Bool dynamic, SCIP_Bool removable, SCIP_Bool stickingatnode)
Definition: cons_setppc.c:9484
SCIP_RETCODE SCIPsetParamsCountsols(SCIP *scip)
SCIP_DECL_DIALOGEXEC(SCIPdialogExecCountPresolve)
SCIP_VAR ** SCIPgetVarsBounddisjunction(SCIP *scip, SCIP_CONS *cons)
SCIP_RETCODE SCIPcount(SCIP *scip)
void SCIPgetCountedSparseSols(SCIP *scip, SCIP_VAR ***vars, int *nvars, SCIP_SPARSESOL ***sols, int *nsols)
SCIP_RETCODE SCIPincludeConshdlrCountsols(SCIP *scip)
SCIP_STAGE SCIPgetStage(SCIP *scip)
Definition: scip_general.c:390
int SCIPgetNContVars(SCIP *scip)
Definition: scip_prob.c:2172
SCIP_VAR ** SCIPgetOrigVars(SCIP *scip)
Definition: scip_prob.c:2405
int SCIPgetNVars(SCIP *scip)
Definition: scip_prob.c:1992
SCIP_RETCODE SCIPaddCons(SCIP *scip, SCIP_CONS *cons)
Definition: scip_prob.c:2770
SCIP_VAR ** SCIPgetVars(SCIP *scip)
Definition: scip_prob.c:1947
int SCIPgetNOrigVars(SCIP *scip)
Definition: scip_prob.c:2432
int SCIPgetNBinVars(SCIP *scip)
Definition: scip_prob.c:2037
void SCIPhashmapFree(SCIP_HASHMAP **hashmap)
Definition: misc.c:3111
int SCIPhashmapGetImageInt(SCIP_HASHMAP *hashmap, void *origin)
Definition: misc.c:3284
SCIP_RETCODE SCIPhashmapCreate(SCIP_HASHMAP **hashmap, BMS_BLKMEM *blkmem, int mapsize)
Definition: misc.c:3077
SCIP_Bool SCIPhashmapExists(SCIP_HASHMAP *hashmap, void *origin)
Definition: misc.c:3426
SCIP_RETCODE SCIPhashmapInsertInt(SCIP_HASHMAP *hashmap, void *origin, int image)
Definition: misc.c:3195
SCIP_RETCODE SCIPdelConsLocal(SCIP *scip, SCIP_CONS *cons)
Definition: scip_prob.c:3475
void SCIPinfoMessage(SCIP *scip, FILE *file, const char *formatstr,...)
Definition: scip_message.c:208
void SCIPverbMessage(SCIP *scip, SCIP_VERBLEVEL msgverblevel, FILE *file, const char *formatstr,...)
Definition: scip_message.c:225
SCIP_MESSAGEHDLR * SCIPgetMessagehdlr(SCIP *scip)
Definition: scip_message.c:88
#define SCIPdebugMsg
Definition: scip_message.h:78
void SCIPdialogMessage(SCIP *scip, FILE *file, const char *formatstr,...)
Definition: scip_message.c:191
void SCIPwarningMessage(SCIP *scip, const char *formatstr,...)
Definition: scip_message.c:120
void SCIPprintError(SCIP_RETCODE retcode)
Definition: scip_general.c:222
SCIP_RETCODE SCIPgetBoolParam(SCIP *scip, const char *name, SCIP_Bool *value)
Definition: scip_param.c:250
SCIP_RETCODE SCIPaddLongintParam(SCIP *scip, const char *name, const char *desc, SCIP_Longint *valueptr, SCIP_Bool isadvanced, SCIP_Longint defaultvalue, SCIP_Longint minvalue, SCIP_Longint maxvalue, SCIP_DECL_PARAMCHGD((*paramchgd)), SCIP_PARAMDATA *paramdata)
Definition: scip_param.c:111
SCIP_Bool SCIPisParamFixed(SCIP *scip, const char *name)
Definition: scip_param.c:219
SCIP_RETCODE SCIPsetIntParam(SCIP *scip, const char *name, int value)
Definition: scip_param.c:487
SCIP_RETCODE SCIPunfixParam(SCIP *scip, const char *name)
Definition: scip_param.c:385
SCIP_RETCODE SCIPsetEmphasis(SCIP *scip, SCIP_PARAMEMPHASIS paramemphasis, SCIP_Bool quiet)
Definition: scip_param.c:882
SCIP_RETCODE SCIPaddBoolParam(SCIP *scip, const char *name, const char *desc, SCIP_Bool *valueptr, SCIP_Bool isadvanced, SCIP_Bool defaultvalue, SCIP_DECL_PARAMCHGD((*paramchgd)), SCIP_PARAMDATA *paramdata)
Definition: scip_param.c:57
SCIP_RETCODE SCIPgetIntParam(SCIP *scip, const char *name, int *value)
Definition: scip_param.c:269
SCIP_RETCODE SCIPsetBoolParam(SCIP *scip, const char *name, SCIP_Bool value)
Definition: scip_param.c:429
SCIP_RETCODE SCIPgetPseudoBranchCands(SCIP *scip, SCIP_VAR ***pseudocands, int *npseudocands, int *npriopseudocands)
Definition: scip_branch.c:733
int SCIPgetNPseudoBranchCands(SCIP *scip)
Definition: scip_branch.c:758
void SCIPconshdlrSetData(SCIP_CONSHDLR *conshdlr, SCIP_CONSHDLRDATA *conshdlrdata)
Definition: cons.c:4227
SCIP_RETCODE SCIPsetConshdlrInit(SCIP *scip, SCIP_CONSHDLR *conshdlr, SCIP_DECL_CONSINIT((*consinit)))
Definition: scip_cons.c:396
int SCIPgetNConshdlrs(SCIP *scip)
Definition: scip_cons.c:965
SCIP_RETCODE SCIPincludeConshdlrBasic(SCIP *scip, SCIP_CONSHDLR **conshdlrptr, const char *name, const char *desc, int enfopriority, int chckpriority, int eagerfreq, SCIP_Bool needscons, SCIP_DECL_CONSENFOLP((*consenfolp)), SCIP_DECL_CONSENFOPS((*consenfops)), SCIP_DECL_CONSCHECK((*conscheck)), SCIP_DECL_CONSLOCK((*conslock)), SCIP_CONSHDLRDATA *conshdlrdata)
Definition: scip_cons.c:181
SCIP_RETCODE SCIPsetConshdlrFree(SCIP *scip, SCIP_CONSHDLR *conshdlr, SCIP_DECL_CONSFREE((*consfree)))
Definition: scip_cons.c:372
SCIP_RETCODE SCIPsetConshdlrEnforelax(SCIP *scip, SCIP_CONSHDLR *conshdlr, SCIP_DECL_CONSENFORELAX((*consenforelax)))
Definition: scip_cons.c:323
int SCIPconshdlrGetNConss(SCIP_CONSHDLR *conshdlr)
Definition: cons.c:4636
const char * SCIPconshdlrGetName(SCIP_CONSHDLR *conshdlr)
Definition: cons.c:4197
SCIP_RETCODE SCIPsetConshdlrExit(SCIP *scip, SCIP_CONSHDLR *conshdlr, SCIP_DECL_CONSEXIT((*consexit)))
Definition: scip_cons.c:420
int SCIPconshdlrGetNEnabledConss(SCIP_CONSHDLR *conshdlr)
Definition: cons.c:4680
SCIP_RETCODE SCIPsetConshdlrCopy(SCIP *scip, SCIP_CONSHDLR *conshdlr, SCIP_DECL_CONSHDLRCOPY((*conshdlrcopy)), SCIP_DECL_CONSCOPY((*conscopy)))
Definition: scip_cons.c:347
SCIP_CONSHDLR * SCIPfindConshdlr(SCIP *scip, const char *name)
Definition: scip_cons.c:941
SCIP_RETCODE SCIPsetConshdlrExitsol(SCIP *scip, SCIP_CONSHDLR *conshdlr, SCIP_DECL_CONSEXITSOL((*consexitsol)))
Definition: scip_cons.c:468
SCIP_RETCODE SCIPsetConshdlrInitsol(SCIP *scip, SCIP_CONSHDLR *conshdlr, SCIP_DECL_CONSINITSOL((*consinitsol)))
Definition: scip_cons.c:444
SCIP_CONSHDLRDATA * SCIPconshdlrGetData(SCIP_CONSHDLR *conshdlr)
Definition: cons.c:4217
int SCIPconshdlrGetNActiveConss(SCIP_CONSHDLR *conshdlr)
Definition: cons.c:4670
SCIP_CONS ** SCIPconshdlrGetConss(SCIP_CONSHDLR *conshdlr)
Definition: cons.c:4593
SCIP_CONSHDLR ** SCIPgetConshdlrs(SCIP *scip)
Definition: scip_cons.c:954
SCIP_Bool SCIPconsIsEnabled(SCIP_CONS *cons)
Definition: cons.c:8311
const char * SCIPconsGetName(SCIP_CONS *cons)
Definition: cons.c:8214
SCIP_RETCODE SCIPreleaseCons(SCIP *scip, SCIP_CONS **cons)
Definition: scip_cons.c:1174
void SCIPdialoghdlrClearBuffer(SCIP_DIALOGHDLR *dialoghdlr)
Definition: dialog.c:446
SCIP_RETCODE SCIPreleaseDialog(SCIP *scip, SCIP_DIALOG **dialog)
Definition: scip_dialog.c:124
SCIP_DIALOG * SCIPdialoghdlrGetRoot(SCIP_DIALOGHDLR *dialoghdlr)
Definition: dialog.c:436
SCIP_Bool SCIPdialogHasEntry(SCIP_DIALOG *dialog, const char *entryname)
Definition: dialog.c:995
SCIP_RETCODE SCIPdialoghdlrAddHistory(SCIP_DIALOGHDLR *dialoghdlr, SCIP_DIALOG *dialog, const char *command, SCIP_Bool escapecommand)
Definition: dialog.c:726
SCIP_RETCODE SCIPincludeDialog(SCIP *scip, SCIP_DIALOG **dialog, SCIP_DECL_DIALOGCOPY((*dialogcopy)), SCIP_DECL_DIALOGEXEC((*dialogexec)), SCIP_DECL_DIALOGDESC((*dialogdesc)), SCIP_DECL_DIALOGFREE((*dialogfree)), const char *name, const char *desc, SCIP_Bool issubmenu, SCIP_DIALOGDATA *dialogdata)
Definition: scip_dialog.c:59
SCIP_RETCODE SCIPaddDialogEntry(SCIP *scip, SCIP_DIALOG *dialog, SCIP_DIALOG *subdialog)
Definition: scip_dialog.c:171
SCIP_RETCODE SCIPdialoghdlrGetWord(SCIP_DIALOGHDLR *dialoghdlr, SCIP_DIALOG *dialog, const char *prompt, char **inputword, SCIP_Bool *endoffile)
Definition: dialog.c:546
SCIP_DIALOG * SCIPgetRootDialog(SCIP *scip)
Definition: scip_dialog.c:157
int SCIPdialogFindEntry(SCIP_DIALOG *dialog, const char *entryname, SCIP_DIALOG **subdialog)
Definition: dialog.c:1028
void SCIPdispLongint(SCIP_MESSAGEHDLR *messagehdlr, FILE *file, SCIP_Longint val, int width)
Definition: disp.c:581
const char * SCIPdispGetName(SCIP_DISP *disp)
Definition: disp.c:335
SCIP_RETCODE SCIPincludeDisp(SCIP *scip, const char *name, const char *desc, const char *header, SCIP_DISPSTATUS dispstatus, SCIP_DECL_DISPCOPY((*dispcopy)), SCIP_DECL_DISPFREE((*dispfree)), SCIP_DECL_DISPINIT((*dispinit)), SCIP_DECL_DISPEXIT((*dispexit)), SCIP_DECL_DISPINITSOL((*dispinitsol)), SCIP_DECL_DISPEXITSOL((*dispexitsol)), SCIP_DECL_DISPOUTPUT((*dispoutput)), SCIP_DISPDATA *dispdata, int width, int priority, int position, SCIP_Bool stripline)
Definition: scip_disp.c:55
SCIP_RETCODE SCIPincludeExternalCodeInformation(SCIP *scip, const char *name, const char *description)
Definition: scip_general.c:744
SCIP_HEUR ** SCIPgetHeurs(SCIP *scip)
Definition: scip_heur.c:271
int SCIPgetNHeurs(SCIP *scip)
Definition: scip_heur.c:282
int SCIPheurGetFreq(SCIP_HEUR *heur)
Definition: heur.c:1538
#define SCIPfreeMemoryArrayNull(scip, ptr)
Definition: scip_mem.h:81
#define SCIPreallocMemoryArray(scip, ptr, newnum)
Definition: scip_mem.h:70
#define SCIPallocMemoryArray(scip, ptr, num)
Definition: scip_mem.h:64
#define SCIPallocBufferArray(scip, ptr, num)
Definition: scip_mem.h:124
#define SCIPreallocBufferArray(scip, ptr, num)
Definition: scip_mem.h:128
#define SCIPfreeBufferArray(scip, ptr)
Definition: scip_mem.h:136
#define SCIPduplicateBufferArray(scip, ptr, source, num)
Definition: scip_mem.h:132
#define SCIPallocBlockMemoryArray(scip, ptr, num)
Definition: scip_mem.h:93
#define SCIPfreeBlockMemory(scip, ptr)
Definition: scip_mem.h:108
#define SCIPfreeBlockMemoryArrayNull(scip, ptr, num)
Definition: scip_mem.h:111
#define SCIPallocBlockMemory(scip, ptr)
Definition: scip_mem.h:89
#define SCIPduplicateBlockMemoryArray(scip, ptr, source, num)
Definition: scip_mem.h:105
SCIP_RETCODE SCIPcheckSolOrig(SCIP *scip, SCIP_SOL *sol, SCIP_Bool *feasible, SCIP_Bool printreason, SCIP_Bool completely)
Definition: scip_sol.c:3305
SCIP_RETCODE SCIPfreeSol(SCIP *scip, SCIP_SOL **sol)
Definition: scip_sol.c:837
SCIP_HEUR * SCIPsolGetHeur(SCIP_SOL *sol)
Definition: sol.c:2804
SCIP_RETCODE SCIPcreateLPSol(SCIP *scip, SCIP_SOL **sol, SCIP_HEUR *heur)
Definition: scip_sol.c:222
SCIP_Real SCIPgetSolVal(SCIP *scip, SCIP_SOL *sol, SCIP_VAR *var)
Definition: scip_sol.c:1213
SCIP_Real SCIPretransformObj(SCIP *scip, SCIP_Real obj)
Definition: scip_sol.c:1428
SCIP_RETCODE SCIPcreatePseudoSol(SCIP *scip, SCIP_SOL **sol, SCIP_HEUR *heur)
Definition: scip_sol.c:312
SCIP_RETCODE SCIPpresolve(SCIP *scip)
Definition: scip_solve.c:2332
SCIP_RETCODE SCIPinterruptSolve(SCIP *scip)
Definition: scip_solve.c:3436
SCIP_RETCODE SCIPsolve(SCIP *scip)
Definition: scip_solve.c:2502
SCIP_Bool SCIPisFeasGE(SCIP *scip, SCIP_Real val1, SCIP_Real val2)
SCIP_Bool SCIPisGE(SCIP *scip, SCIP_Real val1, SCIP_Real val2)
SCIP_Bool SCIPisIntegral(SCIP *scip, SCIP_Real val)
SCIP_Bool SCIPisFeasEQ(SCIP *scip, SCIP_Real val1, SCIP_Real val2)
SCIP_Bool SCIPisFeasLT(SCIP *scip, SCIP_Real val1, SCIP_Real val2)
SCIP_Bool SCIPisFeasLE(SCIP *scip, SCIP_Real val1, SCIP_Real val2)
SCIP_Bool SCIPisFeasIntegral(SCIP *scip, SCIP_Real val)
SCIP_Bool SCIPisGT(SCIP *scip, SCIP_Real val1, SCIP_Real val2)
SCIP_Longint SCIPconvertRealToLongint(SCIP *scip, SCIP_Real real)
SCIP_RETCODE SCIPgetProbvarLinearSum(SCIP *scip, SCIP_VAR **vars, SCIP_Real *scalars, int *nvars, int varssize, SCIP_Real *constant, int *requiredsize, SCIP_Bool mergemultiples)
Definition: scip_var.c:1738
SCIP_RETCODE SCIPvarGetOrigvarSum(SCIP_VAR **var, SCIP_Real *scalar, SCIP_Real *constant)
Definition: var.c:12773
SCIP_Bool SCIPvarIsBinary(SCIP_VAR *var)
Definition: var.c:17598
SCIP_Real SCIPvarGetUbLocal(SCIP_VAR *var)
Definition: var.c:18143
SCIP_Real SCIPvarGetObj(SCIP_VAR *var)
Definition: var.c:17925
SCIP_VARTYPE SCIPvarGetType(SCIP_VAR *var)
Definition: var.c:17583
SCIP_Real SCIPvarGetUbGlobal(SCIP_VAR *var)
Definition: var.c:18087
SCIP_RETCODE SCIPaddVarLocksType(SCIP *scip, SCIP_VAR *var, SCIP_LOCKTYPE locktype, int nlocksdown, int nlocksup)
Definition: scip_var.c:4382
int SCIPvarGetProbindex(SCIP_VAR *var)
Definition: var.c:17767
const char * SCIPvarGetName(SCIP_VAR *var)
Definition: var.c:17418
SCIP_RETCODE SCIPreleaseVar(SCIP *scip, SCIP_VAR **var)
Definition: scip_var.c:1248
SCIP_Bool SCIPvarIsIntegral(SCIP_VAR *var)
Definition: var.c:17609
SCIP_RETCODE SCIPgetNegatedVar(SCIP *scip, SCIP_VAR *var, SCIP_VAR **negvar)
Definition: scip_var.c:1527
SCIP_Real SCIPvarGetLbLocal(SCIP_VAR *var)
Definition: var.c:18133
SCIP_Real SCIPvarGetLbGlobal(SCIP_VAR *var)
Definition: var.c:18077
SCIP_RETCODE SCIPprintVar(SCIP *scip, SCIP_VAR *var, FILE *file)
Definition: scip_var.c:10117
SCIP_RETCODE SCIPgetTransformedVar(SCIP *scip, SCIP_VAR *var, SCIP_VAR **transvar)
Definition: scip_var.c:1439
SCIP_RETCODE SCIPcaptureVar(SCIP *scip, SCIP_VAR *var)
Definition: scip_var.c:1214
void SCIPsortDownPtrPtr(void **ptrarray1, void **ptrarray2, SCIP_DECL_SORTPTRCOMP((*ptrcomp)), int len)
int SCIPsparseSolGetNVars(SCIP_SPARSESOL *sparsesol)
Definition: misc.c:842
SCIP_Longint * SCIPsparseSolGetLbs(SCIP_SPARSESOL *sparsesol)
Definition: misc.c:852
void SCIPsparseSolGetFirstSol(SCIP_SPARSESOL *sparsesol, SCIP_Longint *sol, int nvars)
Definition: misc.c:872
SCIP_RETCODE SCIPsparseSolCreate(SCIP_SPARSESOL **sparsesol, SCIP_VAR **vars, int nvars, SCIP_Bool cleared)
Definition: misc.c:766
SCIP_Longint * SCIPsparseSolGetUbs(SCIP_SPARSESOL *sparsesol)
Definition: misc.c:862
void SCIPsparseSolFree(SCIP_SPARSESOL **sparsesol)
Definition: misc.c:818
SCIP_Bool SCIPsparseSolGetNextSol(SCIP_SPARSESOL *sparsesol, SCIP_Longint *sol, int nvars)
Definition: misc.c:895
int SCIPsnprintf(char *t, int len, const char *s,...)
Definition: misc.c:10880
static const SCIP_Real scalars[]
Definition: lp.c:5743
memory allocation routines
BMS_BLKMEM * SCIPblkmem(SCIP *scip)
Definition: scip_mem.c:57
public methods for managing constraints
public methods for user interface dialog
public methods for displaying runtime statistics
public methods for primal heuristics
public methods for message output
#define SCIPerrorMessage
Definition: pub_message.h:64
#define SCIPdebug(x)
Definition: pub_message.h:93
#define SCIPdebugPrintCons(x, y, z)
Definition: pub_message.h:102
public data structures and miscellaneous methods
methods for sorting joint arrays of various types
public methods for primal CIP solutions
public methods for problem variables
public methods for branching rule plugins and branching
public methods for constraint handler plugins and constraints
public methods for dialog handler plugins
public methods for display handler plugins
general public methods
public methods for primal heuristic plugins and divesets
public methods for memory management
public methods for message handling
public methods for numerical tolerances
public methods for SCIP parameter handling
public methods for global and local (sub)problems
public methods for solutions
public solving methods
public methods for SCIP variables
SCIP_VAR ** vars
Definition: struct_misc.h:48
struct SCIP_ConshdlrData SCIP_CONSHDLRDATA
Definition: type_cons.h:64
@ SCIP_DISPSTATUS_OFF
Definition: type_disp.h:60
@ SCIP_BOUNDTYPE_UPPER
Definition: type_lp.h:57
@ SCIP_BOUNDTYPE_LOWER
Definition: type_lp.h:56
enum SCIP_BoundType SCIP_BOUNDTYPE
Definition: type_lp.h:59
@ SCIP_VERBLEVEL_FULL
Definition: type_message.h:57
@ SCIP_PARAMEMPHASIS_COUNTER
Definition: type_paramset.h:77
@ SCIP_CUTOFF
Definition: type_result.h:48
@ SCIP_FEASIBLE
Definition: type_result.h:45
@ SCIP_INFEASIBLE
Definition: type_result.h:46
enum SCIP_Result SCIP_RESULT
Definition: type_result.h:61
@ SCIP_PLUGINNOTFOUND
Definition: type_retcode.h:54
@ SCIP_OKAY
Definition: type_retcode.h:42
@ SCIP_INVALIDCALL
Definition: type_retcode.h:51
@ SCIP_ERROR
Definition: type_retcode.h:43
enum SCIP_Retcode SCIP_RETCODE
Definition: type_retcode.h:63
@ SCIP_STAGE_PROBLEM
Definition: type_set.h:45
@ SCIP_STAGE_INITPRESOLVE
Definition: type_set.h:48
@ SCIP_STAGE_SOLVED
Definition: type_set.h:54
@ SCIP_STAGE_PRESOLVING
Definition: type_set.h:49
@ SCIP_STAGE_TRANSFORMED
Definition: type_set.h:47
@ SCIP_STAGE_INITSOLVE
Definition: type_set.h:52
@ SCIP_STAGE_EXITPRESOLVE
Definition: type_set.h:50
@ SCIP_STAGE_EXITSOLVE
Definition: type_set.h:55
@ SCIP_STAGE_INIT
Definition: type_set.h:44
@ SCIP_STAGE_FREE
Definition: type_set.h:57
@ SCIP_STAGE_FREETRANS
Definition: type_set.h:56
@ SCIP_STAGE_SOLVING
Definition: type_set.h:53
@ SCIP_STAGE_TRANSFORMING
Definition: type_set.h:46
@ SCIP_STAGE_PRESOLVED
Definition: type_set.h:51
type definitions for symmetry computations
#define SYM_TIMING_AFTERPRESOL
Definition: type_symmetry.h:52
#define SYM_HANDLETYPE_SYMCONS
@ SCIP_VARTYPE_CONTINUOUS
Definition: type_var.h:71
@ SCIP_LOCKTYPE_MODEL
Definition: type_var.h:97