Parent Directory | Revision Log
Reorganising CONOPT binding so that building support for CONOPT is possible without it being installed.
1 | /* ASCEND modelling environment |
2 | Copyright (C) 2006, 2007 Carnegie Mellon University |
3 | |
4 | This program is free software; you can redistribute it and/or modify |
5 | it under the terms of the GNU General Public License as published by |
6 | the Free Software Foundation; either version 2, or (at your option) |
7 | any later version. |
8 | |
9 | This program is distributed in the hope that it will be useful, |
10 | but WITHOUT ANY WARRANTY; without even the implied warranty of |
11 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
12 | GNU General Public License for more details. |
13 | |
14 | You should have received a copy of the GNU General Public License |
15 | along with this program; if not, write to the Free Software |
16 | Foundation, Inc., 59 Temple Place - Suite 330, |
17 | Boston, MA 02111-1307, USA. |
18 | *//** |
19 | @file |
20 | Conditional Modeling Solver (CMSlv) module. |
21 | *//* |
22 | Conditional Modeling Solver |
23 | by Vicente Rico-Ramirez, 04/1997 |
24 | Last in CVS: $Revision: 1.22 $ $Date: 2000/01/25 02:27:58 $ $Author: ballan $ |
25 | */ |
26 | |
27 | #include <math.h> |
28 | |
29 | #include <utilities/config.h> |
30 | #include <utilities/ascConfig.h> |
31 | #include <utilities/ascSignal.h> |
32 | #include <utilities/ascMalloc.h> |
33 | #include <general/tm_time.h> |
34 | #include <utilities/mem.h> |
35 | #include <general/list.h> |
36 | #include <general/mathmacros.h> |
37 | |
38 | #include <linear/mtx_reorder.h> |
39 | |
40 | #include <system/calc.h> |
41 | #include <system/relman.h> |
42 | #include <system/logrelman.h> |
43 | #include <system/bndman.h> |
44 | #include <system/slv_stdcalls.h> |
45 | #include <system/cond_config.h> |
46 | #include <solver/solver.h> |
47 | #include <solver/slvDOF.h> |
48 | |
49 | #include <solver/solver.h> |
50 | |
51 | typedef struct slv9_system_structure *slv9_system_t; |
52 | |
53 | #define SOLVER_CMSLV 9 |
54 | |
55 | ASC_DLLSPEC SolverRegisterFn cmslv_register; |
56 | |
57 | #ifdef ASC_WITH_CONOPT |
58 | # include <solver/conopt_dl.h> |
59 | #else |
60 | # define MAX_INT MAXINT |
61 | # define MAX_REAL MAXDOUBLE |
62 | # define CONOPT_BOUNDLIMIT 1e12 |
63 | #endif |
64 | |
65 | /* |
66 | * definitions to enable/disable the output of partial results in |
67 | * the solution of a problem |
68 | */ |
69 | #define DEBUG FALSE |
70 | #define SHOW_LOGICAL_DETAILS FALSE |
71 | #define SHOW_BOUNDARY_ANALYSIS_DETAILS FALSE |
72 | #define SHOW_OPTIMIZATION_DETAILS FALSE |
73 | #define SHOW_BISECTION_DETAILS FALSE |
74 | #define SHOW_LINEAR_SEARCH_DETAILS FALSE |
75 | #define SHOW_LAGRANGE_DETAILS FALSE |
76 | #define DEBUG_CONSISTENCY FALSE |
77 | #define TEST_CONSISTENCY FALSE |
78 | #define USE_CONSISTENCY FALSE |
79 | |
80 | /* |
81 | * system definitions |
82 | */ |
83 | #define SLV9(s) ((slv9_system_t)(s)) |
84 | #define SERVER (sys->slv) |
85 | #define slv9_PA_SIZE 26 /* MUST INCREMENT WHEN ADDING PARAMETERS */ |
86 | #define LOGSOLVER_OPTION_PTR (sys->parm_array[0]) |
87 | #define LOGSOLVER_OPTION ((*(char **)LOGSOLVER_OPTION_PTR)) |
88 | #define NONLISOLVER_OPTION_PTR (sys->parm_array[1]) |
89 | #define NONLISOLVER_OPTION ((*(char **)NONLISOLVER_OPTION_PTR)) |
90 | #define OPTSOLVER_OPTION_PTR (sys->parm_array[2]) |
91 | #define OPTSOLVER_OPTION ((*(char **)OPTSOLVER_OPTION_PTR)) |
92 | #define TIME_LIMIT_PTR (sys->parm_array[3]) |
93 | #define TIME_LIMIT ((*(int32 *)TIME_LIMIT_PTR)) |
94 | #define ITER_LIMIT_PTR (sys->parm_array[4]) |
95 | #define ITER_LIMIT ((*(int32 *)ITER_LIMIT_PTR)) |
96 | #define ITER_BIS_LIMIT_PTR (sys->parm_array[5]) |
97 | #define ITER_BIS_LIMIT ((*(int32 *)ITER_BIS_LIMIT_PTR)) |
98 | #define TOO_SMALL_PTR (sys->parm_array[6]) |
99 | #define TOO_SMALL ((*(real64 *)TOO_SMALL_PTR)) |
100 | #define LINEAR_SEARCH_FACTOR_PTR (sys->parm_array[7]) |
101 | #define LINEAR_SEARCH_FACTOR ((*(real64 *)LINEAR_SEARCH_FACTOR_PTR)) |
102 | #define SHOW_MORE_IMPT_PTR (sys->parm_array[8]) |
103 | #define SHOW_MORE_IMPT ((*(int32 *)SHOW_MORE_IMPT_PTR)) |
104 | #define SHOW_LESS_IMPT_PTR (sys->parm_array[9]) |
105 | #define SHOW_LESS_IMPT ((*(int32 *)SHOW_LESS_IMPT_PTR)) |
106 | #define AUTO_RESOLVE_PTR (sys->parm_array[10]) |
107 | #define AUTO_RESOLVE ((*(int32 *)AUTO_RESOLVE_PTR)) |
108 | #define UNDEFINED_PTR (sys->parm_array[11]) |
109 | #define UNDEFINED ((*(real64 *)UNDEFINED_PTR)) |
110 | #define DOMLIM_PTR (sys->parm_array[12]) |
111 | #define DOMLIM ((*(int32 *)DOMLIM_PTR)) |
112 | #define OPT_ITER_LIMIT_PTR (sys->parm_array[13]) |
113 | #define OPT_ITER_LIMIT ((*(int32 *)OPT_ITER_LIMIT_PTR)) |
114 | #define INFINITY_PTR (sys->parm_array[14]) |
115 | #define ASC_INFINITY ((*(real64 *)INFINITY_PTR)) |
116 | #define OBJ_TOL_PTR (sys->parm_array[15]) |
117 | #define OBJ_TOL ((*(real64 *)OBJ_TOL_PTR)) |
118 | #define RTMAXJ_PTR (sys->parm_array[16]) |
119 | #define RTMAXJ ((*(real64 *)RTMAXJ_PTR)) |
120 | #define RHO_PTR (sys->parm_array[17]) |
121 | #define RHO ((*(real64 *)RHO_PTR)) |
122 | |
123 | |
124 | /* |
125 | * Client tokens of the different solvers: Conditional, Optimizer, |
126 | * Nonlinear, Logical. We will switch from one client token to |
127 | * another as the solution process occurs. |
128 | */ |
129 | #define NUMBER_OF_CLIENTS 4 |
130 | SlvClientToken token[NUMBER_OF_CLIENTS]; |
131 | int32 solver_index[NUMBER_OF_CLIENTS]; |
132 | |
133 | /* |
134 | * indeces in arrays token and solver_index |
135 | */ |
136 | #define CONDITIONAL_SOLVER 0 |
137 | #define LOGICAL_SOLVER 1 |
138 | #define NONLINEAR_SOLVER 2 |
139 | #define OPTIMIZATION_SOLVER 3 |
140 | |
141 | /* |
142 | * Do we have an optimization problem ?. Global variable initialized |
143 | * to 0 (not optimizing) |
144 | */ |
145 | static int32 g_optimizing = 0; |
146 | |
147 | #if USE_CONSISTENCY |
148 | /* |
149 | * number of subregion visited during the solution of the conditional |
150 | * model. |
151 | */ |
152 | static int32 g_subregions_visited; |
153 | |
154 | #endif /* USE_CONSISTENCY */ |
155 | |
156 | |
157 | /* auxiliar structures */ |
158 | struct boolean_values { |
159 | int32 *pre_val; /* previous values of dis_discrete */ |
160 | int32 *cur_val; /* current values of dis_discrete */ |
161 | }; |
162 | |
163 | struct matching_cases { |
164 | int32 *case_list; /* list of cases */ |
165 | int32 ncases; /* number of cases */ |
166 | int32 diff_subregion; /* subregion ? */ |
167 | }; |
168 | |
169 | struct real_values { |
170 | real64 *pre_values; /* previous values of var_variables */ |
171 | real64 *cur_values; /* current values of var_variables */ |
172 | }; |
173 | |
174 | struct opt_vector { |
175 | real64 *element; /* elements in colum of matrix */ |
176 | }; |
177 | |
178 | struct opt_matrix { |
179 | struct opt_vector *cols; /* columns in matrix */ |
180 | }; |
181 | |
182 | struct subregionID { |
183 | unsigned long ID_number; |
184 | int32 *bool_values; |
185 | }; |
186 | |
187 | struct ds_subregion_list { |
188 | int32 length,capacity; |
189 | struct subregionID *sub_stack; |
190 | }; |
191 | |
192 | struct ds_subregions_visited { |
193 | int32 length,capacity; |
194 | unsigned long *visited; |
195 | }; |
196 | |
197 | /* |
198 | * This solver's data structure (CMSlv) |
199 | */ |
200 | struct slv9_system_structure { |
201 | |
202 | /* |
203 | * Problem definition |
204 | */ |
205 | slv_system_t slv; /* slv_system_t back-link */ |
206 | |
207 | struct rel_relation *obj; /* Objective function: NULL = none */ |
208 | struct var_variable **vlist; /* Variable list (NULL terminated) */ |
209 | struct rel_relation **rlist; /* Relation list (NULL terminated) */ |
210 | struct dis_discrete **dvlist; /* Dis vars list (NULL terminated) */ |
211 | struct logrel_relation **lrlist; /* Logrels list(NULL terminated)*/ |
212 | struct bnd_boundary **blist; /* Variable list (NULL terminated) */ |
213 | struct var_variable **mvlist; |
214 | struct dis_discrete **mdvlist; /* We will not touch the masters list, |
215 | * but they can provide very useful |
216 | * information to the conditional |
217 | * solver since the master index does |
218 | * not change. |
219 | */ |
220 | |
221 | /* |
222 | * for optimization at boundaries |
223 | */ |
224 | struct opt_matrix *coeff_matrix; /* Matrix for optimization problem */ |
225 | struct opt_vector *opt_var_values; /* Values of vars in opt problem */ |
226 | int32 subregions; /* number of subregions at cur bnd */ |
227 | mtx_matrix_t lin_mtx; /* Matrix to define the linear system |
228 | * for calculation of the lagrange |
229 | * multipliers |
230 | */ |
231 | /* |
232 | * For search consistency analysis |
233 | */ |
234 | struct ds_subregion_list subregion_list; |
235 | /* |
236 | * Information about the subregions |
237 | * visited during the solution of the |
238 | * conditional model |
239 | */ |
240 | struct ds_subregions_visited subregions_visited; |
241 | /* |
242 | * ID number of the subregions |
243 | * visited |
244 | */ |
245 | int32 *bool_mindex; /* master indices of boolean vars in |
246 | * the problem associated with WHENs |
247 | */ |
248 | int32 need_consistency_analysis; /* Is the consistency analysis needed */ |
249 | |
250 | |
251 | /* |
252 | * Solver information |
253 | */ |
254 | int32 integrity; /* Has the system been created ? */ |
255 | int32 presolved; /* Has the system been presolved ? */ |
256 | slv_parameters_t p; /* Parameters */ |
257 | slv_status_t s; /* Status (as of iteration end) */ |
258 | int32 cap; /* Order of matrix/vectors */ |
259 | int32 rank; /* Symbolic rank of problem */ |
260 | int32 vused; /* Free and incident variables */ |
261 | int32 vtot; /* length of varlist */ |
262 | int32 mvtot; /* length of master varlist */ |
263 | int32 rused; /* Included relations */ |
264 | int32 rtot; /* length of rellist */ |
265 | real64 clock; /* CPU time */ |
266 | int32 nliter; /* iterations in nonlinear solver */ |
267 | |
268 | void *parm_array[slv9_PA_SIZE]; /* array of pointers to param values */ |
269 | struct slv_parameter pa[slv9_PA_SIZE]; /* &pa[0] => sys->p.parms */ |
270 | |
271 | #ifdef ASC_WITH_CONOPT |
272 | /* |
273 | * Data for optimizer at boundaries (CONOPT) |
274 | */ |
275 | struct conopt_data con; |
276 | #endif |
277 | }; |
278 | |
279 | |
280 | /* |
281 | * Integrity checks |
282 | * ---------------- |
283 | * check_system(sys) |
284 | */ |
285 | |
286 | #define OK ((int32)813029392) |
287 | #define DESTROYED ((int32)103289182) |
288 | |
289 | /* |
290 | * Checks sys for NULL and for integrity. |
291 | */ |
292 | static |
293 | int check_system(slv9_system_t sys){ |
294 | if(sys == NULL ) { |
295 | FPRINTF(ASCERR,"ERROR: (slv9) check_system\n"); |
296 | FPRINTF(ASCERR," NULL system handle.\n"); |
297 | return 1; |
298 | } |
299 | |
300 | switch( sys->integrity ) { |
301 | case OK: |
302 | return 0; |
303 | case DESTROYED: |
304 | FPRINTF(ASCERR,"ERROR: (slv9) check_system\n"); |
305 | FPRINTF(ASCERR," System was recently destroyed.\n"); |
306 | return 1; |
307 | default: |
308 | FPRINTF(ASCERR,"ERROR: (slv9) check_system\n"); |
309 | FPRINTF(ASCERR," System reused or never allocated.\n"); |
310 | return 1; |
311 | } |
312 | } |
313 | |
314 | |
315 | /* |
316 | * General input/output routines |
317 | * ----------------------------- |
318 | * print_var_name(out,sys,var) |
319 | */ |
320 | #define print_var_name(a,b,c) slv_print_var_name((a),(b)->slv,(c)) |
321 | |
322 | |
323 | /* |
324 | * Array operations |
325 | * ---------------- |
326 | * destroy_array(p) |
327 | * create_array(len,type) |
328 | * create_zero_array(len,type) |
329 | */ |
330 | #define destroy_array(p) \ |
331 | if((p) != NULL ) ascfree((p)) |
332 | #define create_array(len,type) \ |
333 | ((len) > 0 ? (type *)ascmalloc((len)*sizeof(type)) : NULL) |
334 | #define create_zero_array(len,type) \ |
335 | ((len) > 0 ? (type *)asccalloc((len),sizeof(type)) : NULL) |
336 | |
337 | |
338 | /* |
339 | * Search Consistency Analysis during iterative process |
340 | * --------------------------------------------------------- |
341 | * The caller of this functions is in charge of |
342 | * defining the extension of the analysis by passing an integer |
343 | * which will tell us if 1) the analysis consider only the current |
344 | * and the previous alternatives or 2) the analysis consider all the |
345 | * alternatives visited at current the state of the solution |
346 | * algorithm. |
347 | */ |
348 | |
349 | /* |
350 | * Handling dynamic allocation of the structural information |
351 | */ |
352 | |
353 | #define alloc_array(nelts,type) \ |
354 | ((nelts) > 0 ? (type *)ascmalloc((nelts)*sizeof(type)) : NULL) |
355 | #define copy_nums(from,too,nnums) \ |
356 | asc_memcpy((from),(too),(nnums)*sizeof(int32)) |
357 | #define copy_subregions(from,too,nsubs) \ |
358 | asc_memcpy((from),(too),(nsubs)*sizeof(struct subregionID)) |
359 | |
360 | #if TEST_CONSISTENCY |
361 | /* |
362 | * Appends the subregion_visited into the list |
363 | */ |
364 | static |
365 | void append_subregion(struct ds_subregion_list *sl, |
366 | struct subregionID sub |
367 | ){ |
368 | if(sl->length == sl->capacity ) { |
369 | int32 newcap; |
370 | struct subregionID *newlist; |
371 | |
372 | newcap = sl->capacity + 10; |
373 | newlist = alloc_array(newcap,struct subregionID); |
374 | copy_subregions((char *)sl->sub_stack,(char *)newlist,sl->length); |
375 | if(sl->sub_stack != NULL ) { |
376 | ascfree(sl->sub_stack); |
377 | } |
378 | sl->sub_stack = newlist; |
379 | sl->capacity = newcap; |
380 | } |
381 | |
382 | sl->sub_stack[sl->length++] = sub; |
383 | } |
384 | |
385 | /* |
386 | * Appends the subregion_visited into the list |
387 | */ |
388 | static |
389 | void append_sub_visited(struct ds_subregions_visited *sv, |
390 | unsigned long sub_visited |
391 | ){ |
392 | if(sv->length == sv->capacity ) { |
393 | int32 newcap; |
394 | unsigned long *newlist; |
395 | |
396 | newcap = sv->capacity + 10; |
397 | newlist = alloc_array(newcap,unsigned long); |
398 | copy_nums((char *)sv->visited,(char *)newlist,sv->length); |
399 | if(sv->visited != NULL ) { |
400 | ascfree(sv->visited); |
401 | } |
402 | sv->visited = newlist; |
403 | sv->capacity = newcap; |
404 | } |
405 | |
406 | sv->visited[sv->length++] = sub_visited; |
407 | } |
408 | |
409 | static |
410 | unsigned long powoftwo (int32 expo){ |
411 | unsigned long val; |
412 | int32 c; |
413 | |
414 | val = 1; |
415 | for (c=1; c<= expo; c++) { |
416 | val = val * 2; |
417 | } |
418 | |
419 | return val; |
420 | } |
421 | |
422 | |
423 | /* |
424 | * Storage information (boolean values) about a subregion so that |
425 | * we can visit it later for interactive strucutral analysis |
426 | */ |
427 | static |
428 | void ID_and_storage_subregion_information(slv_system_t server, |
429 | SlvClientToken asys |
430 | ){ |
431 | slv9_system_t sys; |
432 | struct dis_discrete **bvlist; |
433 | struct dis_discrete *cur_dis; |
434 | struct ds_subregion_list *sl; |
435 | struct ds_subregions_visited *sv; |
436 | struct subregionID *sub; |
437 | dis_filter_t dfilter; |
438 | unsigned long val, visited, sID; |
439 | int32 d, numdvs, numdvf, dcount; |
440 | int32 len, s, found; |
441 | |
442 | sys = SLV9(asys); |
443 | check_system(sys); |
444 | |
445 | bvlist = sys->mdvlist; |
446 | if(bvlist == NULL ) { |
447 | FPRINTF(ASCERR,"ERROR: ID_and_storage_subregion_information.\n"); |
448 | FPRINTF(ASCERR," Master discrete var list was never set.\n"); |
449 | return; |
450 | } |
451 | numdvs = slv_get_num_master_dvars(server); |
452 | |
453 | dfilter.matchbits = (DIS_INWHEN | DIS_BOOLEAN); |
454 | dfilter.matchvalue = (DIS_INWHEN | DIS_BOOLEAN); |
455 | numdvf = slv_count_master_dvars(server,&dfilter); |
456 | |
457 | if(numdvf > 0) { |
458 | sub = (struct subregionID *)(ascmalloc(sizeof(struct subregionID))); |
459 | sub->bool_values = (int32 *)(ascmalloc(numdvf*sizeof(int32))); |
460 | }else{ |
461 | FPRINTF(ASCERR,"ERROR: ID_and_storage_subregion_information.\n"); |
462 | FPRINTF(ASCERR," No boolean variables in the problem \n"); |
463 | return; |
464 | } |
465 | |
466 | dcount = 0; |
467 | val = 0; |
468 | for (d=0; d<numdvs; d++) { |
469 | cur_dis = bvlist[d]; |
470 | if(dis_apply_filter(cur_dis,&dfilter)) { |
471 | sub->bool_values[dcount] = dis_value(cur_dis); |
472 | dcount++; |
473 | if(sub->bool_values[dcount - 1] == 1) { |
474 | val = val + powoftwo(numdvf - dcount); |
475 | } |
476 | } |
477 | } |
478 | if((val == 0 ) && (numdvf > 0) ) { |
479 | val = powoftwo(numdvf); |
480 | } |
481 | sub->ID_number = val; |
482 | #if DEBUG_CONSISTENCY |
483 | FPRINTF(ASCERR,"ID of alternative is %ul \n", val); |
484 | #endif /* DEBUG_CONSISTENCY */ |
485 | visited = val; |
486 | found = 0; |
487 | len = sys->subregions_visited.length; |
488 | if(len > 0) { |
489 | for (s=0; s<len; s++) { |
490 | sID = sys->subregions_visited.visited[s]; |
491 | if(visited == sID) { |
492 | found = 1; |
493 | break; |
494 | } |
495 | } |
496 | } |
497 | |
498 | sv = &(sys->subregions_visited); |
499 | append_sub_visited(sv,visited); |
500 | |
501 | if(found == 0) { |
502 | #if DEBUG_CONSISTENCY |
503 | FPRINTF(ASCERR,"Saving alternative\n"); |
504 | #endif /* DEBUG_CONSISTENCY */ |
505 | sl = &(sys->subregion_list); |
506 | append_subregion(sl,(*sub)); |
507 | }else{ |
508 | destroy_array(sub->bool_values); |
509 | ascfree(sub); |
510 | } |
511 | |
512 | return; |
513 | } |
514 | #endif |
515 | |
516 | /* |
517 | * Destroys subregion information |
518 | */ |
519 | static |
520 | void destroy_subregion_information(SlvClientToken asys){ |
521 | slv9_system_t sys; |
522 | struct subregionID *sub; |
523 | int32 lens, s; |
524 | |
525 | sys = SLV9(asys); |
526 | check_system(sys); |
527 | |
528 | if(sys->subregions_visited.visited != NULL) { |
529 | destroy_array(sys->subregions_visited.visited); |
530 | } |
531 | |
532 | lens = sys->subregion_list.length; |
533 | if(lens != 0) { |
534 | for (s=0; s<lens; s++) { |
535 | sub = &(sys->subregion_list.sub_stack[s]); |
536 | if(sub->bool_values != NULL) { |
537 | destroy_array(sub->bool_values); |
538 | } |
539 | } |
540 | } |
541 | |
542 | if(sys->subregion_list.sub_stack != NULL) { |
543 | destroy_array(sys->subregion_list.sub_stack); |
544 | } |
545 | |
546 | if(sys->bool_mindex != NULL) { |
547 | destroy_array(sys->bool_mindex); |
548 | } |
549 | } |
550 | |
551 | |
552 | #if 0 /** unused function eligible_set_for_neighboring_subregions */ |
553 | /* might be used if DEBUG_CONSISTENCY on */ |
554 | |
555 | /* |
556 | * Storing original values of boolean variables |
557 | */ |
558 | static void store_original_bool_values(struct gl_list_t *bollist, |
559 | struct boolean_values *bval) |
560 | { |
561 | struct dis_discrete *dvar; |
562 | int32 d, dlen; |
563 | |
564 | dlen = gl_length(bollist); |
565 | bval->pre_val = create_array(dlen,int32); |
566 | bval->cur_val = create_array(dlen,int32); |
567 | for (d=1; d<=dlen; d++){ |
568 | dvar = (struct dis_discrete *)gl_fetch(bollist,d); |
569 | bval->cur_val[d-1] = dis_value(dvar); |
570 | bval->pre_val[d-1] = dis_previous_value(dvar); |
571 | } |
572 | } |
573 | |
574 | /* |
575 | * Restoring original values of boolean variables |
576 | */ |
577 | static void restore_original_bool_values(struct gl_list_t *bollist, |
578 | struct boolean_values *bval) |
579 | { |
580 | struct dis_discrete *dvar; |
581 | int32 d, dlen; |
582 | |
583 | dlen = gl_length(bollist); |
584 | for (d=1; d<=dlen; d++){ |
585 | dvar = (struct dis_discrete *)gl_fetch(bollist,d); |
586 | dis_set_boolean_value(dvar,bval->cur_val[d-1]); |
587 | dis_set_value(dvar,bval->cur_val[d-1]); |
588 | dis_set_previous_value(dvar,bval->pre_val[d-1]); |
589 | } |
590 | destroy_array(bval->cur_val); |
591 | destroy_array(bval->pre_val); |
592 | } |
593 | |
594 | #endif /* if 0 */ |
595 | |
596 | /* |
597 | * the first element of cur_cases is in position one. The result is |
598 | * the same array, but ordered and starting in position zero |
599 | */ |
600 | static |
601 | void cases_reorder(int32 *cur_cases, int32 *correct_cases, int32 ncases){ |
602 | int32 cur_case,pos=0,tmp_num,c,ind; |
603 | |
604 | for (c=1; c<=ncases; c++) { |
605 | tmp_num = 0; |
606 | for (ind=1; ind<=ncases; ind++) { |
607 | cur_case = cur_cases[ind]; |
608 | if(tmp_num < cur_case) { |
609 | pos = ind; |
610 | tmp_num = cur_case; |
611 | } |
612 | } |
613 | cur_cases[pos] = 0; |
614 | correct_cases[ncases-c] = tmp_num; |
615 | } |
616 | |
617 | return; |
618 | } |
619 | |
620 | #if 0 /** unused function eligible_set_for_neighboring_subregions */ |
621 | /* might appear if debug_consistency is true. */ |
622 | /* |
623 | * Restoring orignal configuration of the system |
624 | */ |
625 | static void restore_configuration(slv_system_t server, |
626 | struct gl_list_t *bollist) |
627 | |
628 | { |
629 | int32 *cur_cases, *correct_cases; |
630 | int32 ncases; |
631 | |
632 | cur_cases = cases_matching(bollist,&ncases); |
633 | correct_cases = create_array(ncases,int32); |
634 | cases_reorder(cur_cases,correct_cases,ncases); |
635 | set_active_rels_in_subregion(server,correct_cases,ncases,bollist); |
636 | set_active_vars_in_subregion(server); |
637 | destroy_array(cur_cases); |
638 | destroy_array(correct_cases); |
639 | } |
640 | |
641 | |
642 | /* |
643 | * Get the list of boolean variables in the problem which are |
644 | * associated with a WHEN |
645 | */ |
646 | static struct gl_list_t *get_list_of_booleans(slv_system_t server, |
647 | SlvClientToken asys) |
648 | { |
649 | slv9_system_t sys; |
650 | struct dis_discrete **bvlist; |
651 | struct dis_discrete *cur_dis; |
652 | struct gl_list_t *boolvars; |
653 | dis_filter_t dfilter; |
654 | int32 numdvf, numdvs, d, dcount; |
655 | |
656 | sys = SLV9(asys); |
657 | check_system(sys); |
658 | |
659 | bvlist = sys->mdvlist; |
660 | if(bvlist == NULL ) { |
661 | FPRINTF(ASCERR,"ERROR: (slv9) get_list_of_booleans.\n"); |
662 | FPRINTF(ASCERR," Master discrete var list was never set.\n"); |
663 | return NULL; |
664 | } |
665 | numdvs = slv_get_num_master_dvars(server); |
666 | |
667 | dfilter.matchbits = (DIS_INWHEN | DIS_BOOLEAN); |
668 | dfilter.matchvalue = (DIS_INWHEN | DIS_BOOLEAN); |
669 | numdvf = slv_count_master_dvars(server,&dfilter); |
670 | |
671 | if(numdvf == 0) { |
672 | FPRINTF(ASCERR,"ERROR: (slv9) get_list_of_booleans.\n"); |
673 | FPRINTF(ASCERR," No boolean variables in the problem \n"); |
674 | return NULL; |
675 | } |
676 | |
677 | sys->bool_mindex = (int32 *)(ascmalloc(numdvf*sizeof(int32))); |
678 | boolvars = gl_create(numdvf); |
679 | |
680 | dcount = 0; |
681 | for (d=0; d<numdvs; d++) { |
682 | cur_dis = bvlist[d]; |
683 | if(dis_apply_filter(cur_dis,&(dfilter))) { |
684 | gl_append_ptr(boolvars,cur_dis); |
685 | sys->bool_mindex[dcount] = d; |
686 | dcount++; |
687 | } |
688 | } |
689 | |
690 | return boolvars; |
691 | } |
692 | |
693 | #endif /* 0*/ |
694 | |
695 | /* |
696 | * Get the eligible var list for each alternative |
697 | * Return: |
698 | * 1 means everything went right |
699 | * 0 means the analysis has failed with the current parititioning |
700 | * -1 means a memory problem has occurred |
701 | */ |
702 | static |
703 | int32 get_eligible_set(slv_system_t server,struct gl_list_t *disvars, |
704 | int32 *terminate |
705 | ){ |
706 | struct var_variable **vslist; |
707 | struct var_variable **vmlist; |
708 | struct var_variable *mvar, *svar; |
709 | var_filter_t vfilter; |
710 | int32 *cur_cases; |
711 | int32 *correct_cases; |
712 | int32 *vars; |
713 | int32 v, count=0, ind; |
714 | int32 ncases; |
715 | int32 mnum; |
716 | int32 status,dof; |
717 | |
718 | vslist = slv_get_solvers_var_list(server); |
719 | vmlist = slv_get_master_var_list(server); |
720 | mnum = slv_get_num_master_vars(server); |
721 | for (v=0; v<mnum; v++) { |
722 | mvar = vmlist[v]; |
723 | var_set_eligible_in_subregion(mvar,FALSE); |
724 | } |
725 | |
726 | cur_cases = cases_matching(disvars,&ncases); |
727 | correct_cases = create_array(ncases,int32); |
728 | cases_reorder(cur_cases,correct_cases,ncases); |
729 | set_active_rels_in_subregion(server,correct_cases,ncases,disvars); |
730 | set_active_vars_in_subregion(server); |
731 | destroy_array(cur_cases); |
732 | destroy_array(correct_cases); |
733 | |
734 | #if DEBUG_CONSISTENCY |
735 | FPRINTF(ASCERR,"Analyzing alternative:\n"); |
736 | #endif /* DEBUG_CONSISTENCY */ |
737 | |
738 | if(!slvDOF_status(server,(&status),(&dof))) { |
739 | FPRINTF(ASCERR,"ERROR in combinatorial search\n"); |
740 | FPRINTF(ASCERR,"Combinatorial search aborted\n"); |
741 | return -1; |
742 | }else{ |
743 | if(status == 3) { |
744 | #if DEBUG_CONSISTENCY |
745 | FPRINTF(ASCERR,"Alternative is structurally singular\n"); |
746 | #endif /* DEBUG_CONSISTENCY */ |
747 | (*terminate) = 0; |
748 | return 0; |
749 | }else{ |
750 | if(status == 4) { |
751 | #if DEBUG_CONSISTENCY |
752 | FPRINTF(ASCERR,"Alternative is overspecified\n"); |
753 | #endif /* DEBUG_CONSISTENCY */ |
754 | (*terminate) = 0; |
755 | return 0; |
756 | } |
757 | } |
758 | } |
759 | |
760 | if(status == 1) { |
761 | (*terminate) = 0; |
762 | #if DEBUG_CONSISTENCY |
763 | FPRINTF(ASCERR,"Alternative has % d degrees of freedom.\n", dof); |
764 | |
765 | #endif /* DEBUG_CONSISTENCY */ |
766 | if(slvDOF_eligible(server,&(vars))) { |
767 | count = 0; |
768 | while (vars[count] != -1) { |
769 | ind = vars[count]; |
770 | svar = vslist[ind]; |
771 | v = var_mindex(svar); |
772 | mvar = vmlist[v]; |
773 | var_set_eligible_in_subregion(mvar,TRUE); |
774 | count++; |
775 | } |
776 | destroy_array(vars); |
777 | } |
778 | if(dof > count) { |
779 | #if DEBUG_CONSISTENCY |
780 | FPRINTF(ASCERR, |
781 | "Alternative does not have enough number of eligible vars\n"); |
782 | #endif /* DEBUG_CONSISTENCY */ |
783 | return 0; |
784 | } |
785 | } |
786 | |
787 | if(status == 2) { |
788 | #if DEBUG_CONSISTENCY |
789 | FPRINTF(ASCERR,"Alternative is square.\n"); |
790 | #endif /* DEBUG_CONSISTENCY */ |
791 | } |
792 | |
793 | vfilter.matchbits = (VAR_ACTIVE | VAR_INCIDENT | VAR_SVAR |
794 | | VAR_ELIGIBLE_IN_SUBREGION); |
795 | vfilter.matchvalue = (VAR_ACTIVE | VAR_INCIDENT | VAR_SVAR); |
796 | |
797 | for (v=0; v<mnum; v++) { |
798 | mvar = vmlist[v]; |
799 | if(var_apply_filter(mvar,&vfilter)) { |
800 | var_set_eligible(mvar,FALSE); |
801 | } |
802 | var_set_eligible_in_subregion(mvar,FALSE); |
803 | } |
804 | |
805 | return 1; |
806 | } |
807 | |
808 | /* |
809 | * Get the eligible set of variables for each of the alternatives generated |
810 | * by modifying the values of the boolean variables with the values stored |
811 | * during the solution process |
812 | * Return: |
813 | * 1 means everything went right |
814 | * 0 means the analysis has failed with the current partitioning |
815 | * -1 means a memory problem or wierdness has occurred |
816 | */ |
817 | static |
818 | int32 do_search_alternatives(slv_system_t server, SlvClientToken asys, |
819 | struct gl_list_t *disvars, |
820 | int32 *terminate, int32 all_sub |
821 | ){ |
822 | slv9_system_t sys; |
823 | struct dis_discrete *cur_dis; |
824 | struct subregionID *sub; |
825 | int32 *values = NULL; |
826 | int32 dlen, test; |
827 | int32 lens, lenv, v, s, d; |
828 | int32 result; |
829 | unsigned long visited, vID; |
830 | |
831 | |
832 | sys = SLV9(asys); |
833 | |
834 | dlen = gl_length(disvars); |
835 | lenv = sys->subregions_visited.length; |
836 | lens = sys->subregion_list.length; |
837 | |
838 | if(all_sub == 0) { /* current and previous subregion */ |
839 | for (v=lenv-2; v<lenv; v++) { |
840 | test = 0; |
841 | vID = sys->subregions_visited.visited[v]; |
842 | for (s=lens-1; s>=0; s--) { |
843 | sub = &(sys->subregion_list.sub_stack[s]); |
844 | visited = sub->ID_number; |
845 | if(vID == visited) { |
846 | values = sub->bool_values; |
847 | test = 1; |
848 | FPRINTF(ASCERR,"s = %d \n",s); |
849 | break; |
850 | } |
851 | } |
852 | |
853 | if(test == 0) { |
854 | FPRINTF(ASCERR,"ERROR: (slv9) do_search_alternatives \n"); |
855 | FPRINTF(ASCERR," subregion not found \n"); |
856 | return -1; |
857 | } |
858 | #if DEBUG_CONSISTENCY |
859 | FPRINTF(ASCERR,"Alternative = %ul \n", vID); |
860 | #endif /* DEBUG_CONSISTENCY */ |
861 | for (d=0; d<dlen; d++) { |
862 | assert(values != NULL); /* if null, test was 0 above and we returned, in theory */ |
863 | cur_dis = (struct dis_discrete *)(gl_fetch(disvars,d+1)); |
864 | if(values[d] == 1) { |
865 | dis_set_boolean_value(cur_dis,TRUE); |
866 | }else{ |
867 | dis_set_boolean_value(cur_dis,FALSE); |
868 | } |
869 | } |
870 | result = get_eligible_set(server,disvars,terminate); |
871 | if(result != 1) { |
872 | return result; |
873 | } |
874 | } |
875 | |
876 | }else{ /* all visited subregions */ |
877 | |
878 | for (s=lens-1; s>=0; s--) { |
879 | sub = &(sys->subregion_list.sub_stack[s]); |
880 | values = sub->bool_values; |
881 | vID = sub->ID_number; |
882 | #if DEBUG_CONSISTENCY |
883 | FPRINTF(ASCERR,"Alternative = %ul \n", vID); |
884 | #endif /* DEBUG_CONSISTENCY */ |
885 | for (d=0; d<dlen; d++) { |
886 | cur_dis = (struct dis_discrete *)(gl_fetch(disvars,d+1)); |
887 | if(values[d] == 1) { |
888 | dis_set_boolean_value(cur_dis,TRUE); |
889 | }else{ |
890 | dis_set_boolean_value(cur_dis,FALSE); |
891 | } |
892 | } |
893 | result = get_eligible_set(server,disvars,terminate); |
894 | if(result != 1) { |
895 | return result; |
896 | } |
897 | } |
898 | } |
899 | |
900 | return 1; |
901 | } |
902 | |
903 | |
904 | /* |
905 | * Perform consistency analysis for the visited/current-previous subregions. |
906 | * If all_subs is 1, the analysis takes in account all of the subregions |
907 | * visited by the solution algorithm at the current point if the solution |
908 | * procedure. If all_subs is 0, the analysis is only for the current |
909 | * and previous subregion. |
910 | */ |
911 | static |
912 | int32 consistency(slv_system_t server, SlvClientToken asys, |
913 | struct gl_list_t *bollist, |
914 | int32 all_subs, int32 *terminate |
915 | ){ |
916 | slv9_system_t sys; |
917 | struct var_variable **vmlist; |
918 | struct var_variable *mvar; |
919 | var_filter_t vfilter; |
920 | int32 *globeli = NULL; |
921 | int32 dlen; |
922 | int32 mnum, v, elnum; |
923 | int32 result; |
924 | int32 iter; |
925 | |
926 | sys = SLV9(asys); |
927 | check_system(sys); |
928 | |
929 | /* |
930 | * Initializing eligible bit for variables |
931 | */ |
932 | vmlist = slv_get_master_var_list(server); |
933 | mnum = slv_get_num_master_vars(server); |
934 | for (v=0; v<mnum; v++) { |
935 | mvar = vmlist[v]; |
936 | var_set_eligible(mvar,TRUE); |
937 | } |
938 | |
939 | dlen = gl_length(bollist); |
940 | |
941 | #if DEBUG_CONSISTENCY |
942 | FPRINTF(ASCERR,"S e a r c h i n g \n"); |
943 | #endif /* DEBUG_CONSISTENCY */ |
944 | result = do_search_alternatives(server,asys,bollist,terminate,all_subs); |
945 | |
946 | if(result != 1) { |
947 | #if DEBUG_CONSISTENCY |
948 | FPRINTF(ASCERR,"returning failed search after S e a r c h \n"); |
949 | #endif /* DEBUG_CONSISTENCY */ |
950 | return result; |
951 | } |
952 | |
953 | /* |
954 | * Getting the "globally" eligible variables |
955 | */ |
956 | vfilter.matchbits = (VAR_INCIDENT | VAR_SVAR | VAR_ELIGIBLE | VAR_FIXED); |
957 | vfilter.matchvalue = (VAR_INCIDENT | VAR_SVAR | VAR_ELIGIBLE); |
958 | elnum = slv_count_master_vars(server,&vfilter); |
959 | |
960 | if(elnum > 0) { |
961 | globeli = ASC_NEW_ARRAY(int32,elnum); |
962 | elnum = 0; |
963 | for (v=0; v<mnum; v++) { |
964 | mvar = vmlist[v]; |
965 | if(var_apply_filter(mvar,&vfilter)) { |
966 | #if DEBUG_CONSISTENCY |
967 | FPRINTF(ASCERR,"Eligible index = %d \n",v); |
968 | #endif /* DEBUG_CONSISTENCY */ |
969 | globeli[elnum] = v; |
970 | elnum++; |
971 | } |
972 | } |
973 | } |
974 | |
975 | /* |
976 | * Recursively analysis |
977 | */ |
978 | |
979 | if((*terminate) == 1) { |
980 | if(elnum != 0) { |
981 | #if DEBUG_CONSISTENCY |
982 | FPRINTF(ASCERR,"All alternatives are square but the \n"); |
983 | FPRINTF(ASCERR,"Eligible set is not null\n"); |
984 | #endif /* DEBUG_CONSISTENCY */ |
985 | destroy_array(globeli); |
986 | } |
987 | return 1; |
988 | }else{ |
989 | if(elnum == 0) { |
990 | #if DEBUG_CONSISTENCY |
991 | FPRINTF(ASCERR,"No globally eligible variables to be fixed.\n"); |
992 | #endif /* DEBUG_CONSISTENCY */ |
993 | return 0; |
994 | } |
995 | |
996 | for (v=0; v<elnum; v++) { |
997 | iter = 1; |
998 | mvar = vmlist[globeli[v]]; |
999 | var_set_fixed(mvar,TRUE); |
1000 | var_set_potentially_fixed(mvar,TRUE); |
1001 | #if DEBUG_CONSISTENCY |
1002 | FPRINTF(ASCERR,"Fixing index = %d \n",globeli[v]); |
1003 | FPRINTF(ASCERR,"N e s t e d S e a r c h \n"); |
1004 | #endif /* DEBUG_CONSISTENCY */ |
1005 | result = consistency(server,asys,bollist,all_subs,&iter); |
1006 | |
1007 | if(result != 1) { |
1008 | #if DEBUG_CONSISTENCY |
1009 | FPRINTF(ASCERR,"%d eliminated\n",globeli[v]); |
1010 | #endif /* DEBUG_CONSISTENCY */ |
1011 | var_set_fixed(mvar,FALSE); |
1012 | var_set_potentially_fixed(mvar,FALSE); |
1013 | continue; |
1014 | }else{ |
1015 | if(iter == 1) { |
1016 | (*terminate) = 1; |
1017 | #if DEBUG_CONSISTENCY |
1018 | FPRINTF(ASCERR,"%d Acepted \n",globeli[v]); |
1019 | #endif /* DEBUG_CONSISTENCY */ |
1020 | destroy_array(globeli); |
1021 | return 1; |
1022 | }else{ |
1023 | var_set_fixed(mvar,FALSE); |
1024 | var_set_potentially_fixed(mvar,FALSE); |
1025 | continue; |
1026 | } |
1027 | } |
1028 | } |
1029 | destroy_array(globeli); |
1030 | #if DEBUG_CONSISTENCY |
1031 | FPRINTF(ASCERR,"returning 0 after nested search\n"); |
1032 | #endif /* DEBUG_CONSISTENCY */ |
1033 | return 0; |
1034 | } |
1035 | } |
1036 | |
1037 | #if 0 /** unused function eligible_set_for_neighboring_subregions */ |
1038 | /* might appear if debug_consistency is true. */ |
1039 | |
1040 | /* |
1041 | * Get a set of globally eligible variables. Eligible for all the subregions |
1042 | * visited, or for the previous and current subregions. |
1043 | */ |
1044 | static int32 get_globally_eligible(slv_system_t server, SlvClientToken asys, |
1045 | struct gl_list_t *bollist, |
1046 | int32 all_subs, int32 **eliset) |
1047 | { |
1048 | slv9_system_t sys; |
1049 | struct var_variable **vmlist; |
1050 | struct var_variable *mvar; |
1051 | var_filter_t vfilter; |
1052 | int32 dlen; |
1053 | int32 mnum, v, elnum; |
1054 | int32 terminate; |
1055 | int32 result; |
1056 | |
1057 | sys = SLV9(asys); |
1058 | check_system(sys); |
1059 | /* |
1060 | * Initializing eligible bit for variables |
1061 | */ |
1062 | vmlist = slv_get_master_var_list(server); |
1063 | mnum = slv_get_num_master_vars(server); |
1064 | for (v=0; v<mnum; v++) { |
1065 | mvar = vmlist[v]; |
1066 | var_set_eligible(mvar,TRUE); |
1067 | } |
1068 | |
1069 | dlen = gl_length(bollist); |
1070 | |
1071 | /* |
1072 | * initializing |
1073 | */ |
1074 | *eliset = NULL; |
1075 | terminate = 1; |
1076 | |
1077 | #if DEBUG_CONSISTENCY |
1078 | FPRINTF(ASCERR,"S e a r c h i n g \n"); |
1079 | #endif /* DEBUG_CONSISTENCY */ |
1080 | result = do_search_alternatives(server,asys,bollist,&terminate,all_subs); |
1081 | |
1082 | if(result != 1) { |
1083 | if(terminate == 0) { |
1084 | #if DEBUG_CONSISTENCY |
1085 | FPRINTF(ASCERR,"ERROR: some alternatives are either singular or\n"); |
1086 | FPRINTF(ASCERR,"overspecified. All the alternatives have to be\n"); |
1087 | FPRINTF(ASCERR, |
1088 | "either square or underspecified to complete the analysis\n"); |
1089 | #endif /* DEBUG_CONSISTENCY */ |
1090 | } |
1091 | return 0; |
1092 | } |
1093 | |
1094 | /* |
1095 | * Getting the "globally" eligible variables |
1096 | */ |
1097 | vfilter.matchbits = (VAR_INCIDENT | VAR_SVAR | VAR_ELIGIBLE | VAR_FIXED); |
1098 | vfilter.matchvalue = (VAR_INCIDENT | VAR_SVAR | VAR_ELIGIBLE); |
1099 | elnum = slv_count_master_vars(server,&vfilter); |
1100 | |
1101 | *eliset = (int32 *)ascmalloc((elnum+1)*sizeof(int32)); |
1102 | elnum = 0; |
1103 | for (v=0; v<mnum; v++) { |
1104 | mvar = vmlist[v]; |
1105 | if(var_apply_filter(mvar,&vfilter)) { |
1106 | #if DEBUG_CONSISTENCY |
1107 | FPRINTF(ASCERR,"Eligible index = %d \n",v); |
1108 | FPRINTF(ASCERR,"Variable : \n"); |
1109 | print_var_name(ASCERR,sys,mvar); |
1110 | #endif /* DEBUG_CONSISTENCY */ |
1111 | (*eliset)[elnum] = v; |
1112 | elnum++; |
1113 | } |
1114 | } |
1115 | (*eliset)[elnum] = -1; |
1116 | |
1117 | if(elnum == 0) { |
1118 | if(terminate == 0) { |
1119 | #if DEBUG_CONSISTENCY |
1120 | FPRINTF(ASCERR, |
1121 | "Some alternatives are underspecified, but there does\n"); |
1122 | FPRINTF(ASCERR,"not exist a set of eligible variables consistent \n"); |
1123 | FPRINTF(ASCERR,"with all the alternatives\n"); |
1124 | #endif /* DEBUG_CONSISTENCY */ |
1125 | }else{ |
1126 | #if DEBUG_CONSISTENCY |
1127 | FPRINTF(ASCERR,"All alternatives are already square\n"); |
1128 | #endif /* DEBUG_CONSISTENCY */ |
1129 | } |
1130 | return 0; |
1131 | }else{ |
1132 | if(terminate == 1) { |
1133 | #if DEBUG_CONSISTENCY |
1134 | FPRINTF(ASCERR,"All alternatives are square but the \n"); |
1135 | FPRINTF(ASCERR,"Eligible set is not null\n"); |
1136 | #endif /* DEBUG_CONSISTENCY */ |
1137 | } |
1138 | } |
1139 | return 1; |
1140 | } |
1141 | |
1142 | |
1143 | |
1144 | /* |
1145 | * Store and restore values of the boolean variables of the problem |
1146 | * and calls for the the set of globally eligible variables.If all_subs |
1147 | * is 1, the analysis takes in account all of the subregions visited |
1148 | * by the solution algorithm at the current point if the solution |
1149 | * procedure. If all_subs is 0, the analysis is only for the current |
1150 | * and previous subregion. |
1151 | */ |
1152 | static |
1153 | int32 consistent_eligible_set_for_subregions(slv_system_t server, |
1154 | SlvClientToken asys, |
1155 | int32 **vlist, |
1156 | int32 all_subs |
1157 | ){ |
1158 | struct gl_list_t *blist; |
1159 | struct boolean_values bval; |
1160 | int32 result; |
1161 | |
1162 | if(server==NULL || vlist == NULL) { |
1163 | FPRINTF(ASCERR, |
1164 | "consistent_eligible_set_for_subregions called with NULL.\n"); |
1165 | return 0; |
1166 | } |
1167 | |
1168 | blist = get_list_of_booleans(server,asys); |
1169 | |
1170 | if((blist == NULL) || (gl_length(blist) == 0) ) { |
1171 | FPRINTF(ASCERR,"ERROR: (slv9) consistent_eligible_set_for_subregions \n"); |
1172 | FPRINTF(ASCERR," List of boolean vars could not be found\n"); |
1173 | return 0; |
1174 | } |
1175 | |
1176 | store_original_bool_values(blist,&(bval)); |
1177 | result = get_globally_eligible(server,asys,blist,all_subs,vlist); |
1178 | |
1179 | restore_original_bool_values(blist,&(bval)); |
1180 | restore_configuration(server,blist); |
1181 | gl_destroy(blist); |
1182 | |
1183 | if(result == 1) { |
1184 | return 1; |
1185 | }else{ |
1186 | return 0; |
1187 | } |
1188 | |
1189 | } |
1190 | |
1191 | /* |
1192 | * Store and restore values of the boolean variables of the problem |
1193 | * and calls for the consistency analysis of the subregions.If all_subs |
1194 | * is 1, the analysis takes in account all of the subregions visited |
1195 | * by the solution algorithm at the current point if the solution |
1196 | * procedure. If all_subs is 0, the analysis is only for the current |
1197 | * and previous subregion. |
1198 | */ |
1199 | static |
1200 | int32 analyze_subregions(slv_system_t server,SlvClientToken asys, |
1201 | int32 **vlist, int32 all_subs |
1202 | ){ |
1203 | slv9_system_t sys; |
1204 | struct var_variable ** vmlist; |
1205 | struct var_variable *mvar; |
1206 | struct gl_list_t *blist; |
1207 | struct boolean_values bval; |
1208 | var_filter_t vfilter; |
1209 | int32 mnum, elnum, v; |
1210 | int32 result; |
1211 | int32 terminate; |
1212 | |
1213 | sys = SLV9(asys); |
1214 | check_system(sys); |
1215 | |
1216 | if(server==NULL || vlist == NULL) { |
1217 | FPRINTF(ASCERR,"(slv9) analyze_subregions called with NULL.\n"); |
1218 | return 0; |
1219 | } |
1220 | |
1221 | blist = get_list_of_booleans(server,asys); |
1222 | if((blist == NULL) || (gl_length(blist) == 0) ) { |
1223 | FPRINTF(ASCERR,"ERROR: (slv9) analyze_subregions \n"); |
1224 | FPRINTF(ASCERR," List of boolean vars could not be found\n"); |
1225 | return 0; |
1226 | } |
1227 | |
1228 | store_original_bool_values(blist,&(bval)); |
1229 | /* |
1230 | * initializing |
1231 | */ |
1232 | terminate = 1; |
1233 | (*vlist) = NULL; |
1234 | |
1235 | vmlist = slv_get_master_var_list(server); |
1236 | mnum = slv_get_num_master_vars(server); |
1237 | |
1238 | vfilter.matchbits = (VAR_POTENTIALLY_FIXED); |
1239 | vfilter.matchvalue = (VAR_POTENTIALLY_FIXED); |
1240 | |
1241 | result = consistency(server,asys,blist,all_subs,&terminate); |
1242 | |
1243 | if(result == 1) { |
1244 | /* |
1245 | * Getting the set of eligible variables |
1246 | */ |
1247 | elnum = slv_count_master_vars(server,&vfilter); |
1248 | *vlist = (int32 *)ascmalloc((elnum+1)*sizeof(int32)); |
1249 | elnum = 0; |
1250 | for (v=0; v<mnum; v++) { |
1251 | mvar = vmlist[v]; |
1252 | if(var_apply_filter(mvar,&vfilter)) { |
1253 | var_set_fixed(mvar,FALSE); |
1254 | var_set_potentially_fixed(mvar,FALSE); |
1255 | #if DEBUG_CONSISTENCY |
1256 | FPRINTF(ASCERR,"Variable in consistent set: \n"); |
1257 | print_var_name(ASCERR,sys,mvar); |
1258 | #endif /* DEBUG_CONSISTENCY */ |
1259 | (*vlist)[elnum] = v; |
1260 | elnum++; |
1261 | } |
1262 | } |
1263 | (*vlist)[elnum] = -1; |
1264 | |
1265 | restore_original_bool_values(blist,&(bval)); |
1266 | restore_configuration(server,blist); |
1267 | gl_destroy(blist); |
1268 | return 1; |
1269 | }else{ |
1270 | for (v=0; v<mnum; v++) { |
1271 | mvar = vmlist[v]; |
1272 | if(var_apply_filter(mvar,&vfilter)) { |
1273 | var_set_fixed(mvar,FALSE); |
1274 | var_set_potentially_fixed(mvar,FALSE); |
1275 | } |
1276 | } |
1277 | restore_original_bool_values(blist,&(bval)); |
1278 | restore_configuration(server,blist); |
1279 | gl_destroy(blist); |
1280 | return 0; |
1281 | } |
1282 | } |
1283 | |
1284 | |
1285 | /* |
1286 | * Finds the globally eligible set of variables only for the current and |
1287 | * previous subregions |
1288 | */ |
1289 | static |
1290 | int32 eligible_set_for_neighboring_subregions(slv_system_t server, |
1291 | SlvClientToken asys, |
1292 | int32 **vlist |
1293 | ){ |
1294 | slv9_system_t sys; |
1295 | |
1296 | sys = SLV9(asys); |
1297 | check_system(sys); |
1298 | |
1299 | if(sys->mdvlist == NULL ) { |
1300 | FPRINTF(ASCERR,"ERROR: (slv9) eligible_set_for_neighboring_subregions\n"); |
1301 | FPRINTF(ASCERR," Discrete Variable list was never set.\n"); |
1302 | return 0; |
1303 | } |
1304 | |
1305 | if(!(sys->need_consistency_analysis)) { |
1306 | FPRINTF(ASCERR,"Globally eligible set not necessary\n"); |
1307 | FPRINTF(ASCERR,"All the subregions have the same structure \n"); |
1308 | return 0; |
1309 | } |
1310 | |
1311 | if(consistent_eligible_set_for_subregions(server,asys,vlist,0)) { |
1312 | return 1; |
1313 | } |
1314 | |
1315 | return 0; |
1316 | } |
1317 | |
1318 | |
1319 | /* |
1320 | * Perform the consistency analysis algorithm only for the current and |
1321 | * previous subregions |
1322 | */ |
1323 | static |
1324 | int32 consistency_for_neighboring_subregions(slv_system_t server, |
1325 | SlvClientToken asys, |
1326 | int32 **vlist |
1327 | ){ |
1328 | slv9_system_t sys; |
1329 | |
1330 | sys = SLV9(asys); |
1331 | check_system(sys); |
1332 | |
1333 | if(sys->mdvlist == NULL ) { |
1334 | FPRINTF(ASCERR,"ERROR: (slv9) consistency_for_neighboring_subregions\n"); |
1335 | FPRINTF(ASCERR," Discrete Variable list was never set.\n"); |
1336 | return 0; |
1337 | } |
1338 | |
1339 | if(!(sys->need_consistency_analysis)) { |
1340 | FPRINTF(ASCERR,"consistency_analysis is not required\n"); |
1341 | FPRINTF(ASCERR,"All the subregions have the same structure \n"); |
1342 | return 0; |
1343 | } |
1344 | |
1345 | if(analyze_subregions(server,asys,vlist,0)) { |
1346 | return 1; |
1347 | } |
1348 | |
1349 | return 0; |
1350 | } |
1351 | |
1352 | |
1353 | |
1354 | /* |
1355 | * Consistency analysis for visisted subregions. This function |
1356 | * gets the subregions that the solution algorithm has visited and |
1357 | * analyzes them. |
1358 | */ |
1359 | static |
1360 | int32 eligible_set_for_subregions(slv_system_t server, |
1361 | SlvClientToken asys, |
1362 | int32 **vlist |
1363 | ){ |
1364 | slv9_system_t sys; |
1365 | |
1366 | sys = SLV9(asys); |
1367 | check_system(sys); |
1368 | |
1369 | if(sys->mdvlist == NULL ) { |
1370 | FPRINTF(ASCERR,"ERROR: (slv9) eligible_set_for_subregions \n"); |
1371 | FPRINTF(ASCERR," Discrete Variable list was never set.\n"); |
1372 | return 0; |
1373 | } |
1374 | |
1375 | if(!(sys->need_consistency_analysis)) { |
1376 | FPRINTF(ASCERR,"Globally eligible set not necessary \n"); |
1377 | FPRINTF(ASCERR,"All the subregions have the same structure \n"); |
1378 | return 0; |
1379 | } |
1380 | |
1381 | if(consistent_eligible_set_for_subregions(server,asys,vlist,1)) { |
1382 | return 1; |
1383 | } |
1384 | |
1385 | return 0; |
1386 | } |
1387 | |
1388 | |
1389 | /* |
1390 | * Consistency analysis for visisted subregions. This function |
1391 | * gets the subregions that the solution algorithm has visited and |
1392 | * analyzes them. |
1393 | */ |
1394 | static |
1395 | int32 consistency_analysis_for_subregions(slv_system_t server, |
1396 | SlvClientToken asys, |
1397 | int32 **vlist |
1398 | ){ |
1399 | slv9_system_t sys; |
1400 | |
1401 | sys = SLV9(asys); |
1402 | check_system(sys); |
1403 | |
1404 | if(sys->mdvlist == NULL ) { |
1405 | FPRINTF(ASCERR,"ERROR: (slv9) consistency_analysis_for_subregions\n"); |
1406 | FPRINTF(ASCERR," Discrete Variable list was never set.\n"); |
1407 | return 0; |
1408 | } |
1409 | |
1410 | if(!(sys->need_consistency_analysis)) { |
1411 | FPRINTF(ASCERR,"consistency_analysis is not required\n"); |
1412 | FPRINTF(ASCERR,"All the subregions have the same structure \n"); |
1413 | return 0; |
1414 | } |
1415 | |
1416 | if(analyze_subregions(server,asys,vlist,1)) { |
1417 | return 1; |
1418 | } |
1419 | |
1420 | return 0; |
1421 | } |
1422 | |
1423 | #endif /*#if 0 unused functions */ |
1424 | |
1425 | |
1426 | /* |
1427 | * Handling of solution of the Logical Equations |
1428 | * --------------------------------------------------------- |
1429 | * This is made this way because it is a process which will be |
1430 | * required very often. |
1431 | */ |
1432 | |
1433 | /* |
1434 | * Solution of the logical relations encountered in the system based on |
1435 | * the current values of the discrete variables. |
1436 | */ |
1437 | static |
1438 | void solve_logical_relations(slv_system_t server){ |
1439 | slv_set_client_token(server,token[LOGICAL_SOLVER]); |
1440 | slv_set_solver_index(server,solver_index[LOGICAL_SOLVER]); |
1441 | slv_presolve(server); |
1442 | #if SHOW_LOGICAL_DETAILS |
1443 | FPRINTF(ASCERR,"Solving Logical Relations\n"); |
1444 | #endif /* SHOW_LOGICAL_DETAILS */ |
1445 | slv_solve(server); |
1446 | } |
1447 | |
1448 | |
1449 | |
1450 | /* |
1451 | * Handling the modification of parameters in external solvers |
1452 | * --------------------------------------------------------- |
1453 | */ |
1454 | |
1455 | /* |
1456 | * different types of parameter values |
1457 | */ |
1458 | union param_value { |
1459 | int32 i; |
1460 | real64 r; |
1461 | int32 b; |
1462 | char *c; |
1463 | }; |
1464 | |
1465 | |
1466 | /* |
1467 | * Setting the value of a parameter in a subsidiary solver |
1468 | */ |
1469 | static |
1470 | void set_param_in_solver(slv_system_t server, int32 solver, |
1471 | enum parm_type types, char *param, |
1472 | union param_value *value |
1473 | ){ |
1474 | slv_parameters_t p; |
1475 | int32 len,length; |
1476 | SlvClientToken origtoken = slv_get_client_token(server); |
1477 | |
1478 | slv_set_client_token(server,token[solver]); |
1479 | slv_set_solver_index(server,solver_index[solver]); |
1480 | slv_get_parameters(server,&p); |
1481 | length = p.num_parms; |
1482 | for (len = 0; len < length; len++) { |
1483 | if(p.parms[len].type == types) { |
1484 | switch (p.parms[len].type) { |
1485 | case bool_parm: |
1486 | if(strcmp(param,p.parms[len].name) == 0) { |
1487 | p.parms[len].info.b.value = value->b; |
1488 | } |
1489 | break; |
1490 | case real_parm: |
1491 | if(strcmp(param,p.parms[len].name) == 0) { |
1492 | p.parms[len].info.r.value = value->r; |
1493 | } |
1494 | break; |
1495 | case char_parm: |
1496 | if(strcmp(param,p.parms[len].name) == 0) { |
1497 | p.parms[len].info.c.value = value->c; |
1498 | } |
1499 | break; |
1500 | case int_parm: |
1501 | if(strcmp(param,p.parms[len].name) == 0) { |
1502 | p.parms[len].info.i.value = value->i; |
1503 | } |
1504 | break; |
1505 | default: |
1506 | break; |
1507 | } |
1508 | } |
1509 | } |
1510 | |
1511 | /* return to original state */ |
1512 | slv_set_solver_index(server,SOLVER_CMSLV); |
1513 | slv_set_client_token(server,origtoken); |
1514 | |
1515 | return; |
1516 | } |
1517 | |
1518 | |
1519 | /* |
1520 | * Analysis of Discrete Variables |
1521 | * ------------------------------- |
1522 | */ |
1523 | |
1524 | /* |
1525 | * Compare current values of the discrete variables with their previous values |
1526 | * in order to know if some of them have changed. |
1527 | */ |
1528 | static |
1529 | int32 some_dis_vars_changed(slv_system_t server, SlvClientToken asys){ |
1530 | struct dis_discrete **dv, *cur_dis; |
1531 | int32 numdvs, ind; |
1532 | slv9_system_t sys; |
1533 | |
1534 | sys = SLV9(asys); |
1535 | check_system(sys); |
1536 | |
1537 | if(sys->dvlist == NULL ) { |
1538 | FPRINTF(ASCERR,"ERROR: (slv9) some_dis_vars_changed\n"); |
1539 | FPRINTF(ASCERR," Discrete variable list was never set.\n"); |
1540 | return 0; |
1541 | } |
1542 | |
1543 | dv = sys->dvlist; |
1544 | numdvs = slv_get_num_solvers_dvars(server); |
1545 | for( ind = 0; ind < numdvs; ind++ ) { |
1546 | cur_dis = dv[ind]; |
1547 | #if SHOW_LOGICAL_DETAILS |
1548 | FPRINTF(ASCERR,"Boundary index = %d \n",ind); |
1549 | FPRINTF(ASCERR,"Current Value = %d\n",dis_value(cur_dis)); |
1550 | FPRINTF(ASCERR,"Previous Value = %d\n",dis_previous_value(cur_dis)); |
1551 | #endif /* SHOW_LOGICAL_DETAILS */ |
1552 | if((dis_kind(cur_dis)==e_dis_boolean_t ) && dis_inwhen(cur_dis) ) { |
1553 | if(dis_value(cur_dis) != dis_previous_value(cur_dis)) { |
1554 | return 1; |
1555 | } |
1556 | } |
1557 | } |
1558 | return 0; |
1559 | } |
1560 | |
1561 | /* |
1562 | * Compare the original value of a discrete boolean variable (before |
1563 | * perturbation of boundaries) with its values after a solution |
1564 | * of the logical relations with some perturbed values for boundaries. |
1565 | * If those values are different, the bit VAL_MODIFIED is set to |
1566 | * TRUE. This will give us the boolean variable which will change as a |
1567 | * consequence of a boundary crossing. |
1568 | */ |
1569 | static |
1570 | void search_for_modified_dvars(struct dis_discrete **dv, |
1571 | int32 numdvs, |
1572 | struct boolean_values *bval |
1573 | ){ |
1574 | struct dis_discrete *cur_dis; |
1575 | int32 d; |
1576 | int32 orig_value; |
1577 | |
1578 | for (d=0; d<numdvs; d++) { |
1579 | cur_dis = dv[d]; |
1580 | if(dis_inwhen(cur_dis) && dis_boolean(cur_dis)) { |
1581 | orig_value = bval->cur_val[d]; |
1582 | if(orig_value != dis_value(cur_dis)) { |
1583 | dis_set_val_modified(cur_dis,TRUE); |
1584 | } |
1585 | } |
1586 | } |
1587 | } |
1588 | |
1589 | |
1590 | /* |
1591 | * Analysis of Boundaries |
1592 | * ---------------------------- |
1593 | */ |
1594 | |
1595 | /* |
1596 | * Evaluates the current status (satisfied? , at zero?) of a boundary |
1597 | * and sets its flags accordingly. At the same time, it updates the |
1598 | * residual of the relation included in the boundary (see |
1599 | * bndman_calc_satisfied). |
1600 | */ |
1601 | |
1602 | static |
1603 | void update_boundaries(slv_system_t server, SlvClientToken asys){ |
1604 | struct bnd_boundary **bp; |
1605 | int32 numbnds, ind, value; |
1606 | slv9_system_t sys; |
1607 | |
1608 | sys = SLV9(asys); |
1609 | check_system(sys); |
1610 | |
1611 | if(sys->blist == NULL ) { |
1612 | FPRINTF(ASCERR,"ERROR: (slv9) update_boundaries.\n"); |
1613 | FPRINTF(ASCERR," Boundary list was never set.\n"); |
1614 | return; |
1615 | } |
1616 | |
1617 | bp = sys->blist; |
1618 | numbnds = slv_get_num_solvers_bnds(server); |
1619 | |
1620 | for( ind = 0; ind < numbnds; ++ind ) { |
1621 | value = bnd_status_cur(bp[ind]); |
1622 | bnd_set_pre_status(bp[ind],value); |
1623 | value = bndman_calc_satisfied(bp[ind]); |
1624 | bnd_set_cur_status(bp[ind],value); |
1625 | if((bnd_status_cur(bp[ind]) != bnd_status_pre(bp[ind])) && |
1626 | bnd_kind(bp[ind]) == e_bnd_rel && !bnd_at_zero(bp[ind])) { |
1627 | bnd_set_crossed(bp[ind],TRUE); |
1628 | }else{ |
1629 | bnd_set_crossed(bp[ind],FALSE); |
1630 | } |
1631 | if(bnd_kind(bp[ind]) == e_bnd_rel) { |
1632 | value = bndman_calc_at_zero(bp[ind]); |
1633 | bnd_set_at_zero(bp[ind],value); |
1634 | }else{ |
1635 | bnd_set_at_zero(bp[ind],FALSE); |
1636 | } |
1637 | } |
1638 | } |
1639 | |
1640 | |
1641 | /* |
1642 | * Look for some boundary with the CROSSED bit active. If this boundary |
1643 | * is used in some logical relation, the function returns 1, else returns 0 |
1644 | */ |
1645 | static |
1646 | int32 some_boundaries_crossed(slv_system_t server, SlvClientToken asys){ |
1647 | struct bnd_boundary **bp, *cur_bnd; |
1648 | int32 numbnds, ind; |
1649 | slv9_system_t sys; |
1650 | |
1651 | sys = SLV9(asys); |
1652 | check_system(sys); |
1653 | |
1654 | if(sys->blist == NULL ) { |
1655 | FPRINTF(ASCERR,"ERROR: (slv9) some_boundaries_crossed\n"); |
1656 | FPRINTF(ASCERR," Boundary list was never set.\n"); |
1657 | return 0; |
1658 | } |
1659 | |
1660 | bp = sys->blist; |
1661 | numbnds = slv_get_num_solvers_bnds(server); |
1662 | for( ind = 0; ind < numbnds; ++ind ) { |
1663 | cur_bnd = bp[ind]; |
1664 | if(bnd_crossed(cur_bnd) && bnd_in_logrel(cur_bnd)) { |
1665 | return 1; |
1666 | } |
1667 | } |
1668 | return 0; |
1669 | } |
1670 | |
1671 | /* |
1672 | * Look for some boundary with the AT_ZERO bit active.If this boundary |
1673 | * is used in some logical relation, the function returns 1, else returns 0 |
1674 | */ |
1675 | static |
1676 | int32 some_boundaries_at_zero(slv_system_t server, SlvClientToken asys){ |
1677 | struct bnd_boundary **bp, *cur_bnd; |
1678 | int32 numbnds, ind; |
1679 | slv9_system_t sys; |
1680 | |
1681 | sys = SLV9(asys); |
1682 | check_system(sys); |
1683 | |
1684 | if(sys->blist == NULL ) { |
1685 | FPRINTF(ASCERR,"ERROR: (slv9) some_boundaries_at_zero\n"); |
1686 | FPRINTF(ASCERR," Boundary list was never set.\n"); |
1687 | return 0; |
1688 | } |
1689 | |
1690 | bp = sys->blist; |
1691 | numbnds = slv_get_num_solvers_bnds(server); |
1692 | for( ind = 0; ind < numbnds; ++ind ) { |
1693 | cur_bnd = bp[ind]; |
1694 | if(bnd_at_zero(cur_bnd) && bnd_in_logrel(cur_bnd)) { |
1695 | return 1; |
1696 | } |
1697 | } |
1698 | return 0; |
1699 | } |
1700 | |
1701 | /* |
1702 | * Perform the combinatorial perturbation of the boundaries which are |
1703 | * at their zero. That means: We are going to perform a combinatorial |
1704 | * search, changing the truth value of a SATISFIED term (for the |
1705 | * specified boundaries) ON and OFF, and finding the boolean variables |
1706 | * affected for those changes in value of the SATISFIED terms. |
1707 | */ |
1708 | static |
1709 | void do_perturbation_combinations(slv_system_t server, |
1710 | struct boolean_values *bval, |
1711 | struct bnd_boundary **bp, |
1712 | struct dis_discrete **dv, |
1713 | int32 numdvs,int32 *bndatzero, |
1714 | int32 ind, int32 numbndf |
1715 | ){ |
1716 | slv_status_t status; |
1717 | int32 indpo; |
1718 | |
1719 | if(ind<(numbndf-1)) { |
1720 | indpo = ind + 1; |
1721 | bnd_set_perturb(bp[bndatzero[ind]],TRUE); |
1722 | do_perturbation_combinations(server,bval,bp,dv,numdvs, |
1723 | bndatzero,indpo,numbndf); |
1724 | bnd_set_perturb(bp[bndatzero[ind]],FALSE); |
1725 | do_perturbation_combinations(server,bval,bp,dv,numdvs, |
1726 | bndatzero,indpo,numbndf); |
1727 | }else{ |
1728 | if(ind < numbndf) { |
1729 | bnd_set_perturb(bp[bndatzero[ind]],TRUE); |
1730 | solve_logical_relations(server); |
1731 | slv_get_status(server,&status); |
1732 | if(!status.converged) { |
1733 | FPRINTF(ASCERR,"WARNING: \n"); |
1734 | FPRINTF(ASCERR,"(slv9) do_perturbation_combinations\n"); |
1735 | FPRINTF(ASCERR," Not convergence in logical solver \n"); |
1736 | }else{ |
1737 | search_for_modified_dvars(dv,numdvs,bval); |
1738 | } |
1739 | bnd_set_perturb(bp[bndatzero[ind]],FALSE); |
1740 | solve_logical_relations(server); |
1741 | slv_get_status(server,&status); |
1742 | if(!status.converged) { |
1743 | FPRINTF(ASCERR,"WARNING: \n"); |
1744 | FPRINTF(ASCERR,"(slv9) do_perturbation_combinations\n"); |
1745 | FPRINTF(ASCERR," Not convergence in logical solver \n"); |
1746 | }else{ |
1747 | search_for_modified_dvars(dv,numdvs,bval); |
1748 | } |
1749 | }else{ |
1750 | FPRINTF(ASCERR,"ERROR: (slv9) do_perturbation_combinations\n"); |
1751 | FPRINTF(ASCERR," Wrong boundary index as argument\n"); |
1752 | } |
1753 | } |
1754 | return; |
1755 | } |
1756 | |
1757 | |
1758 | /* |
1759 | * Perform the combinatorial search of the subregions. That means: |
1760 | * We perform a combinatorial search, changing the value of the |
1761 | * discrete variables (given in disvars) TRUE and FALSE, and |
1762 | * finding which cases (in the WHENs) applies for each of the |
1763 | * combinations. |
1764 | */ |
1765 | static |
1766 | void do_dvar_values_combinations(struct gl_list_t *disvars, |
1767 | struct matching_cases *cases, |
1768 | int numdvf, int d, |
1769 | int *pos_cases |
1770 | ){ |
1771 | struct dis_discrete *cur_dis; |
1772 | int32 *cur_cases; |
1773 | int32 ncases, dpo; |
1774 | |
1775 | if(d < numdvf) { |
1776 | dpo = d + 1; |
1777 | cur_dis = (struct dis_discrete *)(gl_fetch(disvars,d)); |
1778 | dis_set_boolean_value(cur_dis,TRUE); |
1779 | do_dvar_values_combinations(disvars,cases,numdvf,dpo,pos_cases); |
1780 | dis_set_boolean_value(cur_dis,FALSE); |
1781 | do_dvar_values_combinations(disvars,cases,numdvf,dpo,pos_cases); |
1782 | }else{ |
1783 | if(d == numdvf) { |
1784 | cur_dis = (struct dis_discrete *)(gl_fetch(disvars,d)); |
1785 | dis_set_boolean_value(cur_dis,TRUE); |
1786 | cur_cases = cases_matching(disvars,&ncases); |
1787 | cases[(*pos_cases)].case_list = cur_cases; |
1788 | cases[(*pos_cases)].ncases = ncases; |
1789 | cases[(*pos_cases)].diff_subregion = 1; |
1790 | (*pos_cases)++; |
1791 | dis_set_boolean_value(cur_dis,FALSE); |
1792 | cur_cases = cases_matching(disvars,&ncases); |
1793 | cases[(*pos_cases)].case_list = cur_cases; |
1794 | cases[(*pos_cases)].ncases = ncases; |
1795 | cases[(*pos_cases)].diff_subregion = 1; |
1796 | (*pos_cases)++; |
1797 | }else{ |
1798 | FPRINTF(ASCERR,"ERROR: (slv9) do_dvar_values_combinations\n"); |
1799 | FPRINTF(ASCERR," Wrong discrete var index as argument\n"); |
1800 | } |
1801 | } |
1802 | return; |
1803 | } |
1804 | |
1805 | |
1806 | /* |
1807 | * Orders of the elements of each array of cases, |
1808 | * so that we can compare them easier. |
1809 | */ |
1810 | static |
1811 | void order_case(int32 *case_list, int32 *newcaselist, int ncases){ |
1812 | int32 cur_case,pos=0,tmp_num,c,ind; |
1813 | |
1814 | for (c=1; c<=ncases; c++) { |
1815 | tmp_num = 0; |
1816 | for (ind=1; ind<=ncases; ind++) { |
1817 | cur_case = case_list[ind]; |
1818 | if(tmp_num < cur_case) { |
1819 | pos = ind; |
1820 | tmp_num = cur_case; |
1821 | } |
1822 | } |
1823 | case_list[pos] = 0; |
1824 | newcaselist[ncases-c] = tmp_num; |
1825 | } |
1826 | } |
1827 | |
1828 | |
1829 | |
1830 | /* |
1831 | * Calls for the ordering of the elements of each array of cases, |
1832 | * so that we can compare them easier. |
1833 | */ |
1834 | static |
1835 | void order_cases(struct matching_cases *cases,int pos_cases){ |
1836 | int32 *caselist; |
1837 | int32 cur_ncase,c; |
1838 | int32 *newcaselist; |
1839 | |
1840 | for (c=0; c<pos_cases;c++) { |
1841 | caselist = cases[c].case_list; |
1842 | cur_ncase = cases[c].ncases; |
1843 | if(cur_ncase > 1) { |
1844 | newcaselist = create_array(cur_ncase,int32); |
1845 | order_case(caselist,newcaselist,cur_ncase); |
1846 | cases[c].case_list = newcaselist; |
1847 | destroy_array(caselist); |
1848 | }else{ |
1849 | if(cur_ncase == 1) { |
1850 | newcaselist = create_array(1,int32); |
1851 | newcaselist[0] = caselist[1]; |
1852 | cases[c].case_list = newcaselist; |
1853 | destroy_array(caselist); |
1854 | } |
1855 | } |
1856 | } |
1857 | |
1858 | } |
1859 | |
1860 | |
1861 | |
1862 | /* |
1863 | * Compare two arrays of cases (integer numbers). It returns 1 if they are |
1864 | * equal, else it returns 0. |
1865 | */ |
1866 | static |
1867 | int32 compare_case(int32 *cur_set, int32 *comp_set, int cur_ncases){ |
1868 | int32 cur_case, comp_case, ind; |
1869 | |
1870 | for (ind=0; ind<cur_ncases; ind++) { |
1871 | cur_case = cur_set[ind]; |
1872 | comp_case = comp_set[ind]; |
1873 | if(cur_case != comp_case) { |
1874 | return 0; |
1875 | } |
1876 | } |
1877 | return 1; |
1878 | } |
1879 | |
1880 | |
1881 | /* |
1882 | * Compare the arrays of cases so that we can find the number of |
1883 | * different alternatives (subregions) |
1884 | */ |
1885 | static |
1886 | void compare_cases(struct matching_cases *cases,int pos_cases){ |
1887 | int32 *cur_set, *comp_set, cur_ncases, comp_ncases; |
1888 | int32 c,d; |
1889 | |
1890 | for (c=0; c<pos_cases; c++) { |
1891 | cur_set = cases[c].case_list; |
1892 | cur_ncases = cases[c].ncases; |
1893 | if(cur_ncases == 0) { |
1894 | cases[c].diff_subregion = 0; |
1895 | continue; |
1896 | } |
1897 | for(d=0; d<c; d++) { |
1898 | comp_set = cases[d].case_list; |
1899 | comp_ncases = cases[d].ncases; |
1900 | if(cur_ncases != comp_ncases) { |
1901 | continue; |
1902 | }else{ |
1903 | if(compare_case(cur_set,comp_set,cur_ncases)) { |
1904 | cases[c].diff_subregion = 0; |
1905 | break; |
1906 | } |
1907 | } |
1908 | } |
1909 | } |
1910 | } |
1911 | |
1912 | |
1913 | /* |
1914 | * Finds if my current point lies at a "real" boundary. By "real" |
1915 | * I mean a boundary which really causes a change in the |
1916 | * configuration. It returns 0 if the boundary at zero does not |
1917 | * affect the configuration. If the configuration is affected, |
1918 | * this function will find the number of subregions existing |
1919 | * for the current point as well as the cases (in WHENs) corresponding |
1920 | * to each of the subregions. At the end, the number of subregions is |
1921 | * n_subregions and the cases applying for each of them is stored |
1922 | * in the structure subregions. |
1923 | */ |
1924 | static |
1925 | int32 at_a_boundary(slv_system_t server, SlvClientToken asys, |
1926 | int32 *n_subregions, |
1927 | struct matching_cases **subregions, |
1928 | int32 *cur_subregion, |
1929 | struct gl_list_t *disvars |
1930 | ){ |
1931 | slv9_system_t sys; |
1932 | struct bnd_boundary **bp, *cur_bnd; |
1933 | struct dis_discrete **dv, *cur_dis; |
1934 | struct boolean_values bval; |
1935 | dis_filter_t dfilter; |
1936 | bnd_filter_t bfilter; |
1937 | struct matching_cases *cases; |
1938 | int32 *bndatzero; |
1939 | int32 *dvarmodified; |
1940 | int32 *cur_cases; |
1941 | int32 *caselist, *newcaselist; |
1942 | int32 numbnds, numbndf, b, ind; |
1943 | int32 numdvs, numdvf, d; |
1944 | int32 cur_ncases, assign_cur_sub; |
1945 | int32 pos_cases, comb; |
1946 | char *param; |
1947 | union param_value u; |
1948 | |
1949 | sys = SLV9(asys); |
1950 | check_system(sys); |
1951 | |
1952 | if(sys->blist == NULL ) { |
1953 | FPRINTF(ASCERR,"ERROR: (slv9) at_a_boundary\n"); |
1954 | FPRINTF(ASCERR," Boundary list was never set.\n"); |
1955 | return 0; |
1956 | } |
1957 | |
1958 | if(sys->dvlist == NULL ) { |
1959 | FPRINTF(ASCERR,"ERROR: (slv9) at_a_boundary\n"); |
1960 | FPRINTF(ASCERR," Discrete Variable list was never set.\n"); |
1961 | return 0; |
1962 | } |
1963 | |
1964 | if(!some_boundaries_at_zero(server,asys)) { |
1965 | return 0; |
1966 | } |
1967 | |
1968 | bp = sys->blist; |
1969 | numbnds = slv_get_num_solvers_bnds(server); |
1970 | bfilter.matchbits = (BND_AT_ZERO); |
1971 | bfilter.matchvalue = (BND_AT_ZERO); |
1972 | numbndf = slv_count_solvers_bnds(server,&bfilter); |
1973 | bndatzero = create_array(numbndf,int32); |
1974 | ind = 0; |
1975 | for (b=0; b<numbnds; b++) { |
1976 | cur_bnd = bp[b]; |
1977 | bnd_set_perturb(cur_bnd,FALSE); |
1978 | if(bnd_at_zero(cur_bnd)) { |
1979 | #if SHOW_BOUNDARY_ANALYSIS_DETAILS |
1980 | FPRINTF(ASCERR,"boundary at zero = %d\n",b); |
1981 | #endif /* SHOW_BOUNDARY_ANALYSIS_DETAILS */ |
1982 | bndatzero[ind] = b; |
1983 | ind++; |
1984 | } |
1985 | } |
1986 | |
1987 | dv = sys->dvlist; |
1988 | numdvs = slv_get_num_solvers_dvars(server); |
1989 | bval.cur_val = create_array(numdvs,int32); |
1990 | bval.pre_val = create_array(numdvs,int32); |
1991 | |
1992 | for (d=0; d<numdvs; d++) { |
1993 | cur_dis = dv[d]; |
1994 | dis_set_val_modified(cur_dis,FALSE); |
1995 | bval.cur_val[d] = dis_value(cur_dis); |
1996 | bval.pre_val[d] = dis_previous_value(cur_dis); |
1997 | } |
1998 | |
1999 | #if SHOW_BOUNDARY_ANALYSIS_DETAILS |
2000 | FPRINTF(ASCERR,"Executing combinatorial perturbation of boundaries\n"); |
2001 | #endif /* SHOW_BOUNDARY_ANALYSIS_DETAILS */ |
2002 | |
2003 | /* |
2004 | * Setting the value of the perturbation mode flag in the logical solver |
2005 | * to 1. |
2006 | * PERTURB_BOUNDARY is a boolean parameter of the logical solver |
2007 | * LRSlv. This parameter tells the solver whether it should change |
2008 | * the truth value of the SATISFIED terms or not (only for the |
2009 | * boundaries specified). This trick is important while finding |
2010 | * the number of subregions around a/several boundary(ies). |
2011 | */ |
2012 | param = "perturbboundaries"; |
2013 | u.b = 1; |
2014 | set_param_in_solver(server,LOGICAL_SOLVER,bool_parm,param,&u); |
2015 | |
2016 | ind = 0; |
2017 | do_perturbation_combinations(server,&(bval),bp,dv,numdvs, |
2018 | bndatzero,ind,numbndf); |
2019 | /* |
2020 | * Setting the value of the perturbation mode flag in the logical solver |
2021 | * to 0. |
2022 | */ |
2023 | u.b = 0; |
2024 | set_param_in_solver(server,LOGICAL_SOLVER,bool_parm,param,&u); |
2025 | |
2026 | destroy_array(bndatzero); |
2027 | |
2028 | dfilter.matchbits = (DIS_VAL_MODIFIED); |
2029 | dfilter.matchvalue = (DIS_VAL_MODIFIED); |
2030 | numdvf = slv_count_solvers_dvars(server,&dfilter); |
2031 | |
2032 | if(numdvf == 0) { |
2033 | FPRINTF(ASCERR,"Not really at a boundary\n"); |
2034 | for (d=0; d<numdvs; d++) { |
2035 | cur_dis = dv[d]; |
2036 | dis_set_boolean_value(cur_dis,bval.cur_val[d]); |
2037 | dis_set_value(cur_dis,bval.cur_val[d]); |
2038 | dis_set_previous_value(cur_dis,bval.pre_val[d]); |
2039 | } |
2040 | destroy_array(bval.cur_val); |
2041 | destroy_array(bval.pre_val); |
2042 | return 0; |
2043 | } |
2044 | |
2045 | dvarmodified = create_array(numdvf,int32); |
2046 | ind = 0; |
2047 | for (d=0; d<numdvs; d++) { |
2048 | cur_dis = dv[d]; |
2049 | if(dis_val_modified(cur_dis)) { |
2050 | dvarmodified[ind] = d; |
2051 | gl_append_ptr(disvars,cur_dis); |
2052 | dis_set_val_modified(cur_dis,FALSE); |
2053 | ind++; |
2054 | } |
2055 | } |
2056 | |
2057 | for (d=0; d<numdvs; d++) { |
2058 | cur_dis = dv[d]; |
2059 | dis_set_boolean_value(cur_dis,bval.cur_val[d]); |
2060 | dis_set_value(cur_dis,bval.cur_val[d]); |
2061 | dis_set_previous_value(cur_dis,bval.pre_val[d]); |
2062 | } |
2063 | |
2064 | pos_cases = 1; |
2065 | for (d = 1; d<=numdvf; d++) { |
2066 | pos_cases = pos_cases * 2; |
2067 | } |
2068 | |
2069 | cases = (struct matching_cases *) |
2070 | (ascmalloc((pos_cases)*sizeof(struct matching_cases))); |
2071 | |
2072 | #if SHOW_BOUNDARY_ANALYSIS_DETAILS |
2073 | FPRINTF(ASCERR,"Executing combinatorial search for subregions \n"); |
2074 | #endif /* SHOW_BOUNDARY_ANALYSIS_DETAILS */ |
2075 | |
2076 | d = 1; |
2077 | comb = 0; |
2078 | do_dvar_values_combinations(disvars,cases,numdvf,d,&(comb)); |
2079 | |
2080 | order_cases(cases,pos_cases); |
2081 | compare_cases(cases,pos_cases); |
2082 | |
2083 | (*n_subregions) = 0; |
2084 | for(comb=0; comb<pos_cases;comb++) { |
2085 | if(cases[comb].diff_subregion) { |
2086 | (*n_subregions)++; |
2087 | } |
2088 | } |
2089 | |
2090 | if((*n_subregions)==0) { |
2091 | FPRINTF(ASCERR,"ERROR: at least one subregion must be found\n"); |
2092 | for (d=0; d<numdvs; d++) { |
2093 | cur_dis = dv[d]; |
2094 | dis_set_boolean_value(cur_dis,bval.cur_val[d]); |
2095 | dis_set_value(cur_dis,bval.cur_val[d]); |
2096 | dis_set_previous_value(cur_dis,bval.pre_val[d]); |
2097 | } |
2098 | destroy_array(bval.cur_val); |
2099 | destroy_array(bval.pre_val); |
2100 | for(comb=0; comb<pos_cases;comb++) { |
2101 | destroy_array(cases[comb].case_list); |
2102 | } |
2103 | destroy_array(cases); |
2104 | return 0; |
2105 | } |
2106 | |
2107 | if((*n_subregions)==1) { |
2108 | FPRINTF(ASCERR,"Not really at a boundary\n"); |
2109 | for (d=0; d<numdvs; d++) { |
2110 | cur_dis = dv[d]; |
2111 | dis_set_boolean_value(cur_dis,bval.cur_val[d]); |
2112 | dis_set_value(cur_dis,bval.cur_val[d]); |
2113 | dis_set_previous_value(cur_dis,bval.pre_val[d]); |
2114 | } |
2115 | destroy_array(bval.cur_val); |
2116 | destroy_array(bval.pre_val); |
2117 | for(comb=0; comb<pos_cases;comb++) { |
2118 | destroy_array(cases[comb].case_list); |
2119 | } |
2120 | destroy_array(cases); |
2121 | return 0; |
2122 | } |
2123 | |
2124 | if((*n_subregions) > 0) { |
2125 | (*subregions) = (struct matching_cases *) |
2126 | (ascmalloc(((*n_subregions))*sizeof(struct matching_cases))); |
2127 | (*n_subregions) = 0; |
2128 | for(comb=0; comb<pos_cases;comb++) { |
2129 | if(cases[comb].diff_subregion) { |
2130 | (*subregions)[(*n_subregions)].case_list = cases[comb].case_list; |
2131 | cases[comb].case_list = NULL; |
2132 | (*subregions)[(*n_subregions)].ncases = cases[comb].ncases; |
2133 | cases[comb].ncases = 0; |
2134 | (*subregions)[(*n_subregions)].diff_subregion = 1; |
2135 | (*n_subregions)++; |
2136 | } |
2137 | } |
2138 | } |
2139 | |
2140 | for(comb=0; comb<pos_cases;comb++) { |
2141 | destroy_array(cases[comb].case_list); |
2142 | } |
2143 | destroy_array(cases); |
2144 | |
2145 | |
2146 | assign_cur_sub = 0; |
2147 | /* |
2148 | * Finding the subregion corresponding to the "original" configuration |
2149 | */ |
2150 | for (d=0; d<numdvs; d++) { |
2151 | cur_dis = dv[d]; |
2152 | dis_set_boolean_value(cur_dis,bval.cur_val[d]); |
2153 | dis_set_value(cur_dis,bval.cur_val[d]); |
2154 | dis_set_previous_value(cur_dis,bval.pre_val[d]); |
2155 | } |
2156 | cur_cases = cases_matching(disvars,&cur_ncases); |
2157 | caselist = cur_cases; |
2158 | if(cur_ncases > 1) { |
2159 | newcaselist = create_array(cur_ncases,int32); |
2160 | order_case(caselist,newcaselist,cur_ncases); |
2161 | cur_cases = newcaselist; |
2162 | destroy_array(caselist); |
2163 | }else{ |
2164 | if(cur_ncases == 1) { |
2165 | newcaselist = create_array(1,int32); |
2166 | newcaselist[0] = caselist[1]; |
2167 | cur_cases = newcaselist; |
2168 | destroy_array(caselist); |
2169 | } |
2170 | } |
2171 | for(comb=0; comb<(*n_subregions);comb++) { |
2172 | if((*subregions)[comb].ncases == cur_ncases) { |
2173 | if(compare_case((*subregions)[comb].case_list,cur_cases,cur_ncases)) { |
2174 | (*cur_subregion) = comb; |
2175 | assign_cur_sub = 1; |
2176 | break; |
2177 | } |
2178 | } |
2179 | } |
2180 | |
2181 | if(!assign_cur_sub) { |
2182 | FPRINTF(ASCERR,"PANIC: original configuration not found\n"); |
2183 | } |
2184 | |
2185 | destroy_array(cur_cases); |
2186 | destroy_array(dvarmodified); |
2187 | destroy_array(bval.cur_val); |
2188 | destroy_array(bval.pre_val); |
2189 | return 1; |
2190 | } |
2191 | |
2192 | |
2193 | /* |
2194 | * If some boundary(ies) has been crossed in the iterative scheme, |
2195 | * this function finds the boundary crossed (the first one, if many). |
2196 | * It returns the factor (less than 1) by which the step length has |
2197 | * to be multiplied son that the new point will lie precisely |
2198 | * at that boundary. This factor is found by using the method of |
2199 | * bisection |
2200 | */ |
2201 | static |
2202 | real64 return_to_first_boundary(slv_system_t server, |
2203 | SlvClientToken asys, |
2204 | struct real_values *rvalues, |
2205 | var_filter_t *vfilter |
2206 | ){ |
2207 | slv9_system_t sys; |
2208 | struct bnd_boundary **bp, *cur_bnd; |
2209 | struct var_variable **incidences, **bnd_incidences; |
2210 | struct var_variable *cur_var; |
2211 | bnd_filter_t bfilter; |
2212 | struct boolean_values bval; |
2213 | real64 factor=0.0,fup,flo,newvalue; |
2214 | int32 *bndcrossed; |
2215 | int32 *inc_vars; |
2216 | int32 count,n_incidences,inc,conv_flag,still_crossed; |
2217 | int32 numbnds,numbndf,b,ind; |
2218 | int32 iter,n_iterations; |
2219 | FILE *lif; |
2220 | |
2221 | sys = SLV9(asys); |
2222 | check_system(sys); |
2223 | lif = LIF(sys); |
2224 | |
2225 | if(sys->blist == NULL ) { |
2226 | FPRINTF(ASCERR,"ERROR: (slv9) return_to_first_boundary\n"); |
2227 | FPRINTF(ASCERR," Boundary list was never set.\n"); |
2228 | return 1.0; |
2229 | } |
2230 | |
2231 | if(!some_boundaries_crossed(server,asys)) { |
2232 | return 1.0; |
2233 | } |
2234 | |
2235 | bp = sys->blist; |
2236 | numbnds = slv_get_num_solvers_bnds(server); |
2237 | bfilter.matchbits = (BND_CROSSED); |
2238 | bfilter.matchvalue = (BND_CROSSED); |
2239 | numbndf = slv_count_solvers_bnds(server,&bfilter); |
2240 | bndcrossed = create_array(numbndf,int32); |
2241 | bval.cur_val = create_array(numbndf,int32); |
2242 | bval.pre_val = create_array(numbndf,int32); |
2243 | ind = 0; |
2244 | for (b=0; b<numbnds; b++) { |
2245 | cur_bnd = bp[b]; |
2246 | if(bnd_crossed(cur_bnd)) { |
2247 | bndcrossed[ind] = b; |
2248 | bval.cur_val[ind] = bnd_status_cur(cur_bnd); |
2249 | bval.pre_val[ind] = bnd_status_pre(cur_bnd); |
2250 | ind++; |
2251 | } |
2252 | } |
2253 | |
2254 | count = 0; |
2255 | for (b=0; b<numbndf; b++) { |
2256 | cur_bnd = bp[bndcrossed[b]]; |
2257 | n_incidences = bnd_n_real_incidences(cur_bnd); |
2258 | count = count + n_incidences; |
2259 | } |
2260 | |
2261 | incidences = (struct var_variable **) |
2262 | ( ascmalloc((count)*sizeof(struct var_variable *))); |
2263 | inc_vars = create_array(count,int32); |
2264 | count = 0; |
2265 | for (b=0; b<numbndf; b++) { |
2266 | cur_bnd = bp[bndcrossed[b]]; |
2267 | bnd_incidences = bnd_real_incidence(cur_bnd); |
2268 | n_incidences = bnd_n_real_incidences(cur_bnd); |
2269 | #if SHOW_BOUNDARY_ANALYSIS_DETAILS |
2270 | FPRINTF(lif,"boundary crossed = %d\n",bndcrossed[b]); |
2271 | FPRINTF(lif,"previous boundary status = %d\n",bval.pre_val[b]); |
2272 | FPRINTF(lif,"current boundary status = %d\n",bval.cur_val[b]); |
2273 | #endif /* SHOW_BOUNDARY_ANALYSIS_DETAILS */ |
2274 | for (inc=0; inc<n_incidences; inc++) { |
2275 | incidences[count] = bnd_incidences[inc]; |
2276 | inc_vars[count] = var_mindex(incidences[count]); |
2277 | count++; |
2278 | } |
2279 | } |
2280 | |
2281 | /* bisection to find first boundary crossed */ |
2282 | fup = 1.0; |
2283 | flo = 0.0; |
2284 | conv_flag = 0; |
2285 | iter = 0; |
2286 | |
2287 | /* |
2288 | * Maximum number of bisection iterations. This must be modified |
2289 | * so that it becomes a parameter to be defined by the user |
2290 | */ |
2291 | n_iterations = ITER_BIS_LIMIT; |
2292 | |
2293 | #if SHOW_BOUNDARY_ANALYSIS_DETAILS |
2294 | for (inc=0; inc<count; inc++) { |
2295 | cur_var = incidences[inc]; |
2296 | if(var_apply_filter(cur_var,vfilter)) { |
2297 | FPRINTF(lif,"Variable "); |
2298 | print_var_name(lif,sys,cur_var); PUTC('\n',lif); |
2299 | FPRINTF(lif, |
2300 | "previous value = %f\n",rvalues->pre_values[inc_vars[inc]]); |
2301 | FPRINTF(lif,"current value = %f\n",rvalues->cur_values[inc_vars[inc]]); |
2302 | } |
2303 | } |
2304 | #endif /* SHOW_BOUNDARY_ANALYSIS_DETAILS */ |
2305 | |
2306 | while (conv_flag == 0) { |
2307 | iter++; |
2308 | if(iter>n_iterations) { |
2309 | FPRINTF(ASCERR,"ERROR: (slv9) return_to_first_boundary\n"); |
2310 | FPRINTF(ASCERR,"Could not find the first boundary crossed \n"); |
2311 | FPRINTF(ASCERR,"Returning the last factor calculated\n"); |
2312 | break; |
2313 | } |
2314 | still_crossed = 0; |
2315 | factor = ( fup + flo ) / 2.0; |
2316 | #if SHOW_BISECTION_DETAILS |
2317 | FPRINTF(lif,"fup = %f\n",fup); |
2318 | FPRINTF(lif,"flo = %f\n",flo); |
2319 | FPRINTF(lif,"factor = %f\n",factor); |
2320 | #endif /* SHOW_BISECTION_DETAILS */ |
2321 | for (inc=0; inc<count; inc++) { |
2322 | cur_var = incidences[inc]; |
2323 | if(var_apply_filter(cur_var,vfilter)) { |
2324 | newvalue = rvalues->pre_values[inc_vars[inc]] + factor * |
2325 | ( rvalues->cur_values[inc_vars[inc]] - |
2326 | rvalues->pre_values[inc_vars[inc]] ); |
2327 | var_set_value(cur_var,newvalue); |
2328 | #if SHOW_BISECTION_DETAILS |
2329 | FPRINTF(lif,"Variable "); |
2330 | print_var_name(lif,sys,cur_var); PUTC('\n',lif); |
2331 | FPRINTF(lif,"value after factor = %f\n",newvalue); |
2332 | #endif /* SHOW_BISECTION_DETAILS */ |
2333 | } |
2334 | } |
2335 | |
2336 | update_boundaries(server,asys); |
2337 | for (b=0; b<numbndf; b++) { |
2338 | cur_bnd = bp[bndcrossed[b]]; |
2339 | #if SHOW_BISECTION_DETAILS |
2340 | FPRINTF(lif,"previous status = %d\n", bval.pre_val[b]); |
2341 | FPRINTF(lif,"status aftert factor = %d\n",bnd_status_cur(cur_bnd)); |
2342 | #endif /* SHOW_BISECTION_DETAILS */ |
2343 | if(bnd_status_cur(cur_bnd) != bval.pre_val[b] ) { |
2344 | still_crossed = 1; |
2345 | } |
2346 | } |
2347 | #if SHOW_BISECTION_DETAILS |
2348 | FPRINTF(lif,"still_crossed = %d\n",still_crossed); |
2349 | #endif /* SHOW_BISECTION_DETAILS */ |
2350 | if(still_crossed) { |
2351 | fup = factor; |
2352 | }else{ |
2353 | flo = factor; |
2354 | for (b=0; b<numbndf; b++) { |
2355 | cur_bnd = bp[bndcrossed[b]]; |
2356 | bnd_set_pre_status(cur_bnd,bval.pre_val[b]); |
2357 | bnd_set_cur_status(cur_bnd,bval.pre_val[b]); |
2358 | if(bnd_at_zero(cur_bnd)) { |
2359 | #if SHOW_BOUNDARY_ANALYSIS_DETAILS |
2360 | FPRINTF(ASCERR,"boundary at zero = %d\n",bndcrossed[b]); |
2361 | FPRINTF(lif,"factor = %f\n",factor); |
2362 | for (inc=0; inc<count; inc++) { |
2363 | cur_var = incidences[inc]; |
2364 | if(var_apply_filter(cur_var,vfilter)) { |
2365 | FPRINTF(lif,"Variable "); |
2366 | print_var_name(lif,sys,cur_var); PUTC('\n',lif); |
2367 | FPRINTF(lif,"value after factor = %f\n",var_value(cur_var)); |
2368 | } |
2369 | } |
2370 | #endif /* SHOW_BOUNDARY_ANALYSIS_DETAILS */ |
2371 | conv_flag = 1; |
2372 | } |
2373 | } |
2374 | } |
2375 | } |
2376 | destroy_array(bndcrossed); |
2377 | destroy_array(inc_vars); |
2378 | destroy_array(bval.cur_val); |
2379 | destroy_array(bval.pre_val); |
2380 | destroy_array(incidences); |
2381 | |
2382 | return factor; |
2383 | } |
2384 | |
2385 | |
2386 | /* |
2387 | * Storing values of real variables. |
2388 | * --------------------------------- |
2389 | * |
2390 | * We use the master list of variables since its order does not change |
2391 | * and it is given by the master index. We do not touch the master list, |
2392 | * we only use its order. |
2393 | */ |
2394 | |
2395 | /* |
2396 | * Store the values of the var_variables before a newton-like iteration |
2397 | */ |
2398 | static |
2399 | void store_real_pre_values(slv_system_t server, |
2400 | struct real_values *rvalues |
2401 | ){ |
2402 | struct var_variable **master; |
2403 | struct var_variable *var; |
2404 | int v, vlen; |
2405 | |
2406 | master = slv_get_master_var_list(server); |
2407 | vlen = slv_get_num_master_vars(server); |
2408 | |
2409 | rvalues->pre_values = create_array(vlen,real64); |
2410 | |
2411 | for (v=0; v<vlen; v++) { |
2412 | var = master[v]; |
2413 | rvalues->pre_values[v] = var_value(var); |
2414 | } |
2415 | } |
2416 | |
2417 | /* |
2418 | * Store the values of the var_variables after a newton-like iteration |
2419 | */ |
2420 | static |
2421 | void store_real_cur_values(slv_system_t server, |
2422 | struct real_values *rvalues |
2423 | ){ |
2424 | struct var_variable **master; |
2425 | struct var_variable *var; |
2426 | int v, vlen; |
2427 | |
2428 | master = slv_get_master_var_list(server); |
2429 | vlen = slv_get_num_master_vars(server); |
2430 | |
2431 | rvalues->cur_values = create_array(vlen,real64); |
2432 | |
2433 | for (v=0; v<vlen; v++) { |
2434 | var = master[v]; |
2435 | rvalues->cur_values[v] = var_value(var); |
2436 | } |
2437 | } |
2438 | |
2439 | /* |
2440 | * After the length of the step has been modified so that the current point |
2441 | * lies at a boundary, the values of all the variables is updated so that |
2442 | * they all reduce the length of their step by the same factor. |
2443 | */ |
2444 | static |
2445 | void update_real_var_values(slv_system_t server, |
2446 | struct real_values *rvalues, |
2447 | var_filter_t *vfilter, real64 factor |
2448 | ){ |
2449 | struct var_variable **master; |
2450 | struct var_variable *var; |
2451 | real64 newvalue; |
2452 | int v, vlen; |
2453 | |
2454 | master = slv_get_master_var_list(server); |
2455 | vlen = slv_get_num_master_vars(server); |
2456 | |
2457 | for (v=0; v<vlen; v++) { |
2458 | var = master[v]; |
2459 | if(var_apply_filter(var,vfilter)) { |
2460 | newvalue = rvalues->pre_values[v] + |
2461 | factor * (rvalues->cur_values[v] - rvalues->pre_values[v]); |
2462 | var_set_value(var,newvalue); |
2463 | } |
2464 | } |
2465 | destroy_array(rvalues->cur_values); |
2466 | destroy_array(rvalues->pre_values); |
2467 | } |
2468 | |
2469 | |
2470 | /* |
2471 | * Set the flagbit NONBASIC for all the variables in the list |
2472 | * to the value passed as argument |
2473 | */ |
2474 | static |
2475 | void set_nonbasic_status_in_var_list(slv_system_t server, |
2476 | uint32 value |
2477 | ){ |
2478 | struct var_variable **master; |
2479 | struct var_variable *var; |
2480 | int v, vlen; |
2481 | |
2482 | master = slv_get_master_var_list(server); |
2483 | vlen = slv_get_num_master_vars(server); |
2484 | |
2485 | for (v=0; v<vlen; v++) { |
2486 | var = master[v]; |
2487 | var_set_nonbasic(var,value); |
2488 | } |
2489 | } |
2490 | |
2491 | |
2492 | /* |
2493 | * After the length of the step has been modified so that the current point |
2494 | * lies at a boundary, the residuals of the equations are updated. |
2495 | */ |
2496 | static void update_relations_residuals(slv_system_t server) |
2497 | { |
2498 | struct rel_relation **master; |
2499 | struct rel_relation *rel; |
2500 | rel_filter_t rfilter; |
2501 | real64 resid; |
2502 | int32 r, rlen, status; |
2503 | |
2504 | master = slv_get_master_rel_list(server); |
2505 | rlen = slv_get_num_master_rels(server); |
2506 | |
2507 | rfilter.matchbits = (REL_INCLUDED | REL_EQUALITY); |
2508 | rfilter.matchvalue = (REL_INCLUDED | REL_EQUALITY); |
2509 | |
2510 | #ifdef ASC_SIGNAL_TRAPS |
2511 | Asc_SignalHandlerPush(SIGFPE,SIG_IGN); |
2512 | #endif |
2513 | |
2514 | for (r=0; r<rlen; r++) { |
2515 | rel = master[r]; |
2516 | if(rel_apply_filter(rel,&rfilter)) { |
2517 | resid = relman_eval(rel,&status,1); |
2518 | } |
2519 | } |
2520 | #ifdef ASC_SIGNAL_TRAPS |
2521 | Asc_SignalHandlerPop(SIGFPE,SIG_IGN); |
2522 | #endif |
2523 | |
2524 | } |
2525 | |
2526 | |
2527 | #ifdef ASC_WITH_CONOPT |
2528 | /*------------------------------------------------------------------------------ |
2529 | CALLBACK ROUTINES FOR CONOPT |
2530 | */ |
2531 | |
2532 | /* |
2533 | * COIRMS Based on the information provided in Coispz, CONOPT will |
2534 | * allocate the number of vectors into which the user can define |
2535 | * the details of the model. The details of the model are defined |
2536 | * here. |
2537 | * |
2538 | * COIRMS(lower, curr, upper, vsta, type,rhs, fv, esta, colsta, |
2539 | * rowno, value, nlflag, n, m, nz, usrmem) |
2540 | * |
2541 | * lower - lower bounds on the variables |
2542 | * curr - intial values of the variables |
2543 | * upper - upper bounds on the variables |
2544 | * vsta - initial status of the variable(o nonbasic, 1 basic) |
2545 | * type - types of equations (0 equality,1 greater,2 less) |
2546 | * rhs - values of the right hand sides |
2547 | * fv - sum of the nonlinear terms in the initial point |
2548 | * esta - initial status of the slack in the constraint (nonbasic,basic) |
2549 | * colsta- start of column pointers |
2550 | * rowno - row or equation numbers of the nonzeros |
2551 | * value - values of the jacobian elements |
2552 | * nlflag- nonlinearity flags(0 nonzero constant,1 varying) |
2553 | * n - number of variables |
2554 | * m - number of constraints |
2555 | * nz - number of jacobian elements |
2556 | * usrmem- user memory defined by conopt |
2557 | */ |
2558 | static |
2559 | int COI_CALL slv9_conopt_readmatrix( |
2560 | double *lower, double *curr, double *upper |
2561 | , int *vsta, int *type, double *rhs |
2562 | , int *esta, int *colsta, int *rowno |
2563 | , double *value, int *nlflag, int *n, int *m, int *nz |
2564 | , double *usrmem |
2565 | ){ |
2566 | slv9_system_t sys; |
2567 | struct var_variable *var; |
2568 | struct var_variable **varlist; |
2569 | struct opt_matrix *coeff_matrix; |
2570 | real64 /*obj_val,*/ deriv; |
2571 | real64 nominal, up, low, uplow; |
2572 | int32 num_var, n_subregions, c, num_eqns; |
2573 | int32 numnz, eq; |
2574 | int32 count, totvar; |
2575 | double limit; |
2576 | |
2577 | static var_filter_t vfilter = { |
2578 | VAR_ACTIVE_AT_BND | VAR_INCIDENT | VAR_SVAR | VAR_FIXED |
2579 | ,VAR_ACTIVE_AT_BND | VAR_INCIDENT | VAR_SVAR | 0 |
2580 | }; |
2581 | |
2582 | UNUSED_PARAMETER(vsta); |
2583 | UNUSED_PARAMETER(esta); |
2584 | |
2585 | sys = (slv9_system_t)usrmem; |
2586 | n_subregions = sys->subregions; |
2587 | coeff_matrix = sys->coeff_matrix; |
2588 | num_var = (*n) - n_subregions; |
2589 | num_eqns = num_var + 1; |
2590 | |
2591 | varlist = sys->mvlist; |
2592 | totvar = sys->mvtot; |
2593 | |
2594 | /* fetch the configured bound from solver parameters */ |
2595 | limit = ASC_INFINITY; |
2596 | |
2597 | /* CONSOLE_DEBUG("Got limit value of %g",limit); */ |
2598 | |
2599 | /* |
2600 | * Variables: Current Value, lower value and upper value. Note that for |
2601 | * this problem the variables are the vector of steps dx. So we "invent" |
2602 | * approximations for the bounds based on the bounds of the real |
2603 | * variables of the problem. The use of the parameter ASC_INFINITY is |
2604 | * Hack to keep CONOPT from complaining for large bounds. |
2605 | */ |
2606 | |
2607 | count = 0; |
2608 | for (c=0; c<totvar; c++) { |
2609 | var = varlist[c]; |
2610 | if(var_apply_filter(var,&vfilter)) { |
2611 | nominal = var_nominal(var); |
2612 | low = var_lower_bound(var); |
2613 | up = var_upper_bound(var); |
2614 | uplow = fabs( up - low); |
2615 | |
2616 | if(-uplow > -limit){ |
2617 | lower[count] = -uplow; |
2618 | }else{ |
2619 | lower[count] = -0.5*limit; |
2620 | /* CONSOLE_DEBUG("Reducing lower bound limit for var %d to %e",count,lower[count]); */ |
2621 | } |
2622 | |
2623 | if(uplow < limit){ |
2624 | upper[count] = uplow; |
2625 | }else{ |
2626 | upper[count] = 0.5*limit; |
2627 | /* CONSOLE_DEBUG("Reducing upper bound limit for var %d to %e",count,upper[count]); */ |
2628 | } |
2629 | |
2630 | curr[count] = 0.5 * nominal; |
2631 | count++; |
2632 | } |
2633 | } |
2634 | /* alphas for each subregion */ |
2635 | for (c=count; c<(*n); c++) { |
2636 | lower[c] = 0.0; |
2637 | upper[c] = 1.0; |
2638 | curr[c] = 1.0; |
2639 | } |
2640 | |
2641 | /*CONSOLE_DEBUG("ALL BOUNDS:"); |
2642 | for(c=0;c<(*n);++c){ |
2643 | fprintf(stderr,"%d: lower = %g, upper = %g\n",c,lower[c],upper[c]); |
2644 | }*/ |
2645 | |
2646 | /* |
2647 | * vsta not in use since STATOK, ipsz[14], is zero |
2648 | */ |
2649 | |
2650 | /* |
2651 | * All the equations, but the last row (which is the objective), are |
2652 | * equalities. |
2653 | */ |
2654 | for (c = 0; c < (*m); c++) { |
2655 | type[c] = 0; |
2656 | } |
2657 | type[(*m)-1] = 3; |
2658 | |
2659 | |
2660 | /* |
2661 | * RHS. It is zero for all the equations except the summation of |
2662 | * alphas, whose RHS is one. |
2663 | */ |
2664 | for (c = 0; c < (*m); c++) { |
2665 | rhs[c] = 0; |
2666 | } |
2667 | rhs[(*m)-2] = 1.0; |
2668 | |
2669 | #ifdef DISUSED_CONOPT_PARAMETER |
2670 | /* |
2671 | * fv =0 for all linear relations. For the objective is the two |
2672 | * norm |
2673 | */ |
2674 | for (c = 0; c < (*m); c++) { |
2675 | fv[c] = 0; |
2676 | } |
2677 | obj_val = 0.0; |
2678 | for (c = 0; c<num_var; c++) { |
2679 | obj_val = obj_val + (curr[c] * curr[c]); |
2680 | } |
2681 | fv[(*m)-1] = obj_val; |
2682 | #endif |
2683 | |
2684 | /* |
2685 | * esta not used since STATOK is zero |
2686 | */ |
2687 | |
2688 | |
2689 | /* |
2690 | * For the following parameters, it is important ot see that: |
2691 | * 1) The values for the rows and nonzeros that conopt wants start |
2692 | * with 1, not with 0. |
2693 | * 2) The indeces of the arrays that we are using in the C side start |
2694 | * with 0. |
2695 | */ |
2696 | |
2697 | /* |
2698 | * colsta |
2699 | */ |
2700 | |
2701 | for (c=0; c<num_var; c++) { |
2702 | colsta[c] = 2 * c; |
2703 | } |
2704 | |
2705 | for (c=num_var; c<(*n); c++) { |
2706 | colsta[c] = 2 * num_var + num_eqns * (c - num_var); |
2707 | } |
2708 | |
2709 | colsta[*n] = *nz; /** @TODO check this */ |
2710 | |
2711 | /* |
2712 | rowno, value and nlflag can be done in same loop. The use of the |
2713 | parameter RTMAXJ is really a Hack to keep CONOPT from complaining |
2714 | about large derivatives |
2715 | */ |
2716 | |
2717 | numnz = 0; |
2718 | for (c=0; c<num_var; c++) { |
2719 | rowno[numnz] = c; |
2720 | nlflag[numnz] = 0; |
2721 | value[numnz] = -1; |
2722 | numnz++; |
2723 | rowno[numnz] = *m - 1; |
2724 | nlflag[numnz] = 1; |
2725 | numnz++; |
2726 | } |
2727 | |
2728 | for (c=num_var; c<(*n); c++) { |
2729 | numnz = 2 * num_var + num_eqns * (c - num_var); |
2730 | for(eq = 0; eq<num_eqns-1; eq++) { |
2731 | rowno[numnz] = eq; |
2732 | nlflag[numnz] = 0; |
2733 | deriv = -1.0 * (coeff_matrix->cols[c - num_var].element[eq]); |
2734 | if(deriv > RTMAXJ ) { |
2735 | deriv = 0.5 * RTMAXJ; |
2736 | }else{ |
2737 | if(deriv < -RTMAXJ ) { |
2738 | deriv = -0.5*RTMAXJ; |
2739 | } |
2740 | } |
2741 | value[numnz] = deriv; |
2742 | numnz++; |
2743 | } |
2744 | rowno[numnz] = num_eqns - 1; |
2745 | nlflag[numnz] = 0; |
2746 | value[numnz] = 1.0; |
2747 | } |
2748 | |
2749 | return 0; |
2750 | } |
2751 | |
2752 | #if 0 /* not in API any more */ |
2753 | /* |
2754 | * COIFBL Defines the nonlinearities of the model by returning |
2755 | * numerical values. It works on a block of rows during each call. |
2756 | * COIFBL( x, g, otn, nto, from, to, jac, stcl, rnum, cnum, nl, strw, |
2757 | * llen, indx, mode, errcnt, n, m, n1, m1, nz, usrmem) |
2758 | * |
2759 | * x - punt of evaluation provided by conopt |
2760 | * g - vector of function values |
2761 | * otn - old to new permutation vector |
2762 | * nto - new to old permutation vector |
2763 | * from - range in permutation |
2764 | * to - range in permutation |
2765 | * jac - vector of jacobian values. |
2766 | * The following are vectors defining the jacobian structure |
2767 | * stcl - start of column pointers |
2768 | * rnum - row numbers |
2769 | * cnum - column numbers |
2770 | * nl - nonlinearity flags |
2771 | * strw - start row pointers |
2772 | * llen - count of linear jacobian elements |
2773 | * indx - pointers from the row-wise representation |
2774 | * mode - indicator of mode of evaluation |
2775 | * errcnt- number of function evaluation errors |
2776 | * n - umber of variables |
2777 | * m - number of constraints |
2778 | * n1 - n+1 |
2779 | * m1 - m+1 |
2780 | * nz - number of jacobian elements |
2781 | * usrmem- user memory defined by conopt |
2782 | */ |
2783 | static void slv9_coifbl(real64 *x, real64 *g, int32 *otn, int32 *nto, |
2784 | int32 *from, int32 *to, real64 *jac, int32 *stcl, |
2785 | int32 *rnum, int32 *cnum, int32 *nl, int32 *strw, |
2786 | int32 *llen, int32 *indx, int32 *mode, int32 *errcnt, |
2787 | int32 *n, int32 *m, int32 *n1, int32 *m1, |
2788 | int32 *nz, real64 *usrmem) |
2789 | { |
2790 | /* non defined for this solver */ |
2791 | |
2792 | /* stop gcc whining about unused parameter */ |
2793 | (void)x; (void)g; (void)otn; (void)nto; (void)from; (void)to; |
2794 | (void)jac; (void)stcl; (void)rnum; (void)cnum; (void)nl; (void)strw; |
2795 | (void)llen; (void)indx; (void)mode; (void)errcnt; (void)n; (void)m; |
2796 | (void)n1; (void)m1; (void)nz; |
2797 | (void)usrmem; |
2798 | |
2799 | return; |
2800 | } |
2801 | #endif |
2802 | |
2803 | /* |
2804 | * COIFDE Defines the nonlinearities of the model by returning |
2805 | * numerical values. It works on one row or equation at a time |
2806 | * COIFDE(x, g, jac, rowno, jcnm, mode, errcnt, newpt, n, nj, usrmem) |
2807 | * |
2808 | * x - punt of evaluation provided by conopt |
2809 | * g - function value |
2810 | * jac - jacobian values |
2811 | * rowno - number of the row for which nonlinearities will be eval |
2812 | * jcnm - list of column number fon the NL nonzeros |
2813 | * mode - indicator of mode of evaluation |
2814 | * errcnt - sum of number of func evaluation errors thus far |
2815 | * newpt - new point indicator |
2816 | * nj - number of nonlinear nonzero jacobian elements |
2817 | * n - number of variables |
2818 | * usrmem - user memory |
2819 | * |
2820 | * For the optimization problem at a boundary, this subroutine will |
2821 | * be called only of the objective function, constraint number m. |
2822 | * |
2823 | */ |
2824 | static |
2825 | int COI_CALL slv9_conopt_fdeval( |
2826 | double *x, double *g, double *jac |
2827 | , int *rowno, int *jcnm, int *mode, int *ignerr |
2828 | , int *errcnt, int *newpt, int *n, int *nj |
2829 | , double *usrmem |
2830 | ){ |
2831 | slv9_system_t sys; |
2832 | int32 num_vars, v; |
2833 | real64 obj, deriv; |
2834 | |
2835 | UNUSED_PARAMETER(jcnm); |
2836 | UNUSED_PARAMETER(errcnt); |
2837 | UNUSED_PARAMETER(newpt); |
2838 | UNUSED_PARAMETER(n); |
2839 | UNUSED_PARAMETER(nj); |
2840 | |
2841 | sys = (slv9_system_t)usrmem; |
2842 | num_vars = sys->con.n - sys->subregions; |
2843 | |
2844 | if(*mode == 1 || *mode == 3) { |
2845 | if(*rowno == sys->con.m - 1){ |
2846 | obj = 0.0; |
2847 | for (v=0; v<num_vars; v++) { |
2848 | obj = obj + (x[v] * x[v]); |
2849 | } |
2850 | *g = obj; |
2851 | }else{ |
2852 | ERROR_REPORTER_HERE(ASC_PROG_ERR,"Wrong number of constraints"); |
2853 | return 1; |
2854 | } |
2855 | } |
2856 | |
2857 | /* |
2858 | * The use of the parameter RTMAXJ is really a Hack to keep CONOPT |
2859 | * from complaining about large derivatives. |
2860 | */ |
2861 | |
2862 | if(*mode == 2 || *mode == 3) { |
2863 | if(*rowno == sys->con.m - 1){ |
2864 | for (v=0; v<num_vars; v++) { |
2865 | deriv = 2.0 * x[v]; |
2866 | if(deriv > RTMAXJ ) { |
2867 | deriv = 0.5*RTMAXJ; |
2868 | }else{ |
2869 | if(deriv < -RTMAXJ ) { |
2870 | deriv = -0.5*RTMAXJ; |
2871 | } |
2872 | } |
2873 | jac[v] = deriv; |
2874 | } |
2875 | }else{ |
2876 | ERROR_REPORTER_HERE(ASC_PROG_ERR,"Wrong number of constraints"); |
2877 | return 1; |
2878 | } |
2879 | } |
2880 | |
2881 | return 0; |
2882 | } |
2883 | |
2884 | |
2885 | /* |
2886 | * COISTA Pass the solution from CONOPT to the modeler. It returns |
2887 | * completion status |
2888 | * COISTA(modsta, solsts, iter, objval, usrmem) |
2889 | * |
2890 | * modsta - model status |
2891 | * solsta - solver status |
2892 | * iter - number of iterations |
2893 | * objval - objective value |
2894 | * usrmem - user memory |
2895 | */ |
2896 | static |
2897 | int COI_CALL slv9_conopt_status(int *modsta, int *solsta, int *iter |
2898 | , double *objval, double *usrmem |
2899 | ){ |
2900 | slv9_system_t sys; |
2901 | |
2902 | sys = (slv9_system_t)usrmem; |
2903 | |
2904 | sys->con.modsta = *modsta; |
2905 | sys->con.solsta = *solsta; |
2906 | sys->con.iter = *iter; |
2907 | sys->con.obj = *objval; |
2908 | |
2909 | return 0; |
2910 | } |
2911 | |
2912 | /** |
2913 | CONOPT error message reporting |
2914 | */ |
2915 | int COI_CALL slv9_conopt_errmsg( int* ROWNO, int* COLNO, int* POSNO, int* MSGLEN |
2916 | , double* USRMEM, char* MSG, int LENMSG |
2917 | ){ |
2918 | slv9_system_t sys; |
2919 | char *varname=NULL; |
2920 | struct var_variable **vp; |
2921 | |
2922 | sys = (slv9_system_t)USRMEM; |
2923 | |
2924 | |
2925 | if(*COLNO!=-1){ |
2926 | vp=sys->mvlist; |
2927 | vp = vp + *COLNO; |
2928 | assert(*vp!=NULL); |
2929 | varname= var_make_name(SERVER,*vp); |
2930 | } |
2931 | |
2932 | ERROR_REPORTER_START_NOLINE(ASC_PROG_ERR); |
2933 | if(*ROWNO == -1){ |
2934 | FPRINTF(ASCERR,"Variable %d (Maybe it's '%s'): ",*COLNO,varname); |
2935 | ASC_FREE(varname); |
2936 | }else if(*COLNO == -1 ){ |
2937 | FPRINTF(ASCERR,"Relation %d: ",*ROWNO); |
2938 | }else{ |
2939 | FPRINTF(ASCERR,"Variable %d (Maybe it's '%s') appearing in relation %d: ",*COLNO,varname,*ROWNO); |
2940 | ASC_FREE(varname); |
2941 | } |
2942 | FPRINTF(ASCERR,"%*s", *MSGLEN, MSG); |
2943 | error_reporter_end_flush(); |
2944 | return 0; |
2945 | } |
2946 | |
2947 | |
2948 | /* |
2949 | * COIRS Pass the solution from CONOPT to the modeler. It returns |
2950 | * solution values |
2951 | * COIRS(val, xmar, xbas, xsta, yval, ymar, ybas, ysta, n, m, usrmem) |
2952 | * |
2953 | * xval - the solution values of the variables |
2954 | * xmar - corresponding marginal values |
2955 | * xbas - basis indicator for column (at bound, basic, nonbasic) |
2956 | * xsta - status of column (normal, nonoptimal, infeasible,unbounded) |
2957 | * yval - values of the left hand side in all the rows |
2958 | * ymar - corresponding marginal values |
2959 | * ybas - basis indicator for row |
2960 | * ysta - status of row |
2961 | * n - number of variables |
2962 | * m - number of constraints |
2963 | * usrmem - user memory |
2964 | */ |
2965 | static |
2966 | int COI_CALL slv9_conopt_solution(double *xval, double *xmar, int *xbas, int *xsta, |
2967 | double *yval, double *ymar, int *ybas, int * ysta, |
2968 | int *n, int *m, double *usrmem |
2969 | ){ |
2970 | slv9_system_t sys; |
2971 | struct opt_vector *opt_var_values; |
2972 | int32 c; |
2973 | real64 value; |
2974 | |
2975 | UNUSED_PARAMETER(xmar);UNUSED_PARAMETER(xbas);UNUSED_PARAMETER(xsta); |
2976 | UNUSED_PARAMETER(yval);UNUSED_PARAMETER(ymar);UNUSED_PARAMETER(ybas); |
2977 | UNUSED_PARAMETER(ysta);UNUSED_PARAMETER(m); |
2978 | |
2979 | sys = (slv9_system_t)usrmem; |
2980 | opt_var_values = sys->opt_var_values; |
2981 | |
2982 | for (c = 0; c < (*n); c++) { |
2983 | value = xval[c]; |
2984 | opt_var_values->element[c] = value; |
2985 | } |
2986 | |
2987 | return 0; |
2988 | } |
2989 | |
2990 | #if 0 |
2991 | /* |
2992 | * COIUSZ communicates and update of an existing model to CONOPT |
2993 | * COIUSZ(nintg, ipsz, nreal, rpsz, usrmem) |
2994 | * |
2995 | * nintg - number of positions in ipsz |
2996 | * ipsz - array describing problem size and options |
2997 | * nreal - number of positions in rpsz |
2998 | * rpsz - array of reals describing problem size and options |
2999 | * usrmem- user memory |
3000 | */ |
3001 | static void slv9_coiusz(int32 *nintg, int32 *ipsz, int32 *nreal, real64 *rpsz, |
3002 | real64 *usrmem) |
3003 | { |
3004 | /* non defined for this solver */ |
3005 | |
3006 | /* |
3007 | * stop gcc whining about unused parameter |
3008 | */ |
3009 | (void)nintg; (void)ipsz; (void)nreal; (void)rpsz; |
3010 | (void)usrmem; |
3011 | |
3012 | return; |
3013 | } |
3014 | #endif |
3015 | |
3016 | /* |
3017 | * COIOPT communicates non-default option values to CONOPT |
3018 | * COIOPT(name, rval, ival, lval, usrmem) |
3019 | * name - the name of a CONOPT CR-cell defined by the modeler |
3020 | * rval - the value to be assigned to name if the cells contains a real |
3021 | * ival - the value to be assigned to name if the cells contains an int |
3022 | * lval - the value to be assigned to name if the cells contains a log value |
3023 | * usrmem - user memory |
3024 | */ |
3025 | static |
3026 | int COI_CALL slv9_conopt_option( |
3027 | int *NCALL, double *rval, int *ival, int *logical |
3028 | , double *usrmem, char *name, int lenname |
3029 | ){ |
3030 | slv9_system_t sys; |
3031 | sys = (slv9_system_t)usrmem; |
3032 | |
3033 | UNUSED_PARAMETER(logical); |
3034 | |
3035 | name = memset(name,' ',8); |
3036 | while (sys->con.opt_count < slv9_PA_SIZE) { |
3037 | if(strlen(sys->p.parms[sys->con.opt_count].interface_label) == 6) { |
3038 | if(strncmp(sys->p.parms[sys->con.opt_count].interface_label, |
3039 | "R",1) == 0) { |
3040 | name = strncpy(name, sys->p.parms[sys->con.opt_count]. /* . break */ |
3041 | interface_label,6); |
3042 | *rval = sys->p.parms[sys->con.opt_count].info.r.value; |
3043 | sys->con.opt_count++; |
3044 | return 0; |
3045 | } else if(strncmp(sys->p.parms[sys->con.opt_count]. /* . break */ |
3046 | interface_label,"L",1) == 0) { |
3047 | name = strncpy(name,sys->p.parms[sys->con.opt_count]. /* . break */ |
3048 | interface_label,6); |
3049 | *ival = sys->p.parms[sys->con.opt_count].info.i.value; |
3050 | sys->con.opt_count++; |
3051 | return 0; |
3052 | } |
3053 | } |
3054 | sys->con.opt_count++; |
3055 | } |
3056 | |
3057 | /* sending blank to quit iterative calling */ |
3058 | name = memset(name,' ',8); |
3059 | return 0; |
3060 | } |
3061 | |
3062 | #if 0 /* see slv_conopt_iterate */ |
3063 | /* |
3064 | * COIPSZ communicates the model size and structure to CONOPT |
3065 | * COIPSZ(nintgr, ipsz, nreal, rpsz, usrmem) |
3066 | * |
3067 | * ningtr - number of positions in ipsz |
3068 | * ipsz - array describing problem size and options |
3069 | * nreal - number of positions in rpsz |
3070 | * rpsz - array of reals describing problem size and options |
3071 | * usrmem - user memory |
3072 | */ |
3073 | static void slv9_coipsz(int32 *nintg, int32 *ipsz, int32 *nreal, real64 *rpsz, |
3074 | real64 *usrmem) |
3075 | { |
3076 | slv9_system_t sys; |
3077 | |
3078 | /* |
3079 | * stop gcc whining about unused parameter |
3080 | */ |
3081 | (void)nintg; (void)nreal; |
3082 | |
3083 | sys = (slv9_system_t)usrmem; |
3084 | /* |
3085 | * Integer array |
3086 | */ |
3087 | ipsz[F2C(1)] = sys->con.n; /* variables */ |
3088 | ipsz[F2C(2)] = sys->con.m; /* constraints including objective */ |
3089 | ipsz[F2C(3)] = sys->con.nz; /* non zeros in Jacobian */ |
3090 | ipsz[F2C(4)] = sys->con.nz - (sys->con.m - 2); /* linear nonzeros */ |
3091 | ipsz[F2C(5)] = sys->con.m - 2; /* nonlinear nonzeros */ |
3092 | ipsz[F2C(6)] = -1; /* direction of optimization min */ |
3093 | ipsz[F2C(7)] = sys->con.m; /* objective will be last row */ |
3094 | ipsz[F2C(8)] = OPT_ITER_LIMIT; /* iteration limit */ |
3095 | ipsz[F2C(9)] = DOMLIM; /* max number of error in func evals */ |
3096 | ipsz[F2C(10)] = 0; /* output to file */ |
3097 | ipsz[F2C(11)] = 1; /* progress info to screen */ |
3098 | ipsz[F2C(12)] = 1; /* correct value of func in coirms */ |
3099 | ipsz[F2C(13)] = 0; /* not correct value of jacs in coirms */ |
3100 | ipsz[F2C(14)] = 0; /* status not known by modeler */ |
3101 | ipsz[F2C(15)] = 0; /* function value include only NL terms */ |
3102 | ipsz[F2C(16)] = 1; /* Objective is a constraint */ |
3103 | ipsz[F2C(17)] = 0; /* sorted order for jacobian */ |
3104 | ipsz[F2C(18)] = 0; /* append the log file after restarts */ |
3105 | ipsz[F2C(19)] = 0; /* one subroutine call to coirms */ |
3106 | ipsz[F2C(20)] = 0; /* eval subroutine is coifde */ |
3107 | ipsz[F2C(21)] = 0; /* no debugging of derivatives */ |
3108 | ipsz[F2C(22)] = 0; /* coifde not called for linear eqns */ |
3109 | /* |
3110 | * skipping remainder of ipsz which are fortran io parameters |
3111 | */ |
3112 | |
3113 | /* |
3114 | * Real array |
3115 | */ |
3116 | rpsz[F2C(1)] = ASC_INFINITY; /* infinity */ |
3117 | rpsz[F2C(2)] = -ASC_INFINITY; /* -infinity */ |
3118 | rpsz[F2C(3)] = UNDEFINED; /* undefined */ |
3119 | rpsz[F2C(6)] = 0; /* work space allocated by conopt */ |
3120 | rpsz[F2C(7)] = TIME_LIMIT; /* resource limit (time) */ |
3121 | rpsz[F2C(8)] = 1; /* initial value for vars if none given */ |
3122 | |
3123 | |
3124 | } |
3125 | #endif |
3126 | |
3127 | |
3128 | /** |
3129 | Perform CONOPT solution. For the details of what this does in the larger |
3130 | context of CMSlv, read (???) |
3131 | |
3132 | @TODO document this. |
3133 | |
3134 | @see conopt.h |
3135 | */ |
3136 | static |
3137 | void slv_conopt_iterate(slv9_system_t sys){ |
3138 | |
3139 | if(sys->con.cntvect == NULL){ |
3140 | sys->con.cntvect = ASC_NEW_ARRAY(int,COIDEF_Size()); |
3141 | } |
3142 | |
3143 | COIDEF_Ini(sys->con.cntvect); |
3144 | |
3145 | /* |
3146 | We pass pointer to sys as usrmem data. |
3147 | Cast back to slv9_system_t to access the information required |
3148 | */ |
3149 | COIDEF_UsrMem(sys->con.cntvect,(double *)sys); |
3150 | |
3151 | COIDEF_NumVar(sys->con.cntvect, &(sys->con.n)); |
3152 | COIDEF_NumCon(sys->con.cntvect, &(sys->con.m)); /* include the obj fn */ |
3153 | COIDEF_NumNZ(sys->con.cntvect, &(sys->con.nz)); |
3154 | COIDEF_NumNlNz(sys->con.cntvect, &(sys->con.nlnz)); |
3155 | COIDEF_OptDir(sys->con.cntvect, &(sys->con.optdir)); |
3156 | |
3157 | COIDEF_ObjCon(sys->con.cntvect, &(sys->con.objcon)); /* objective will be last row */ |
3158 | COIDEF_Base(sys->con.cntvect, &(sys->con.base)); |
3159 | COIDEF_ErrLim(sys->con.cntvect, &(DOMLIM)); |
3160 | COIDEF_ItLim(sys->con.cntvect, &(OPT_ITER_LIMIT)); |
3161 | |
3162 | COIDEF_ReadMatrix(sys->con.cntvect, &slv9_conopt_readmatrix); |
3163 | COIDEF_FDEval(sys->con.cntvect, &slv9_conopt_fdeval); |
3164 | COIDEF_Option(sys->con.cntvect, &slv9_conopt_option); |
3165 | COIDEF_Solution(sys->con.cntvect, &slv9_conopt_solution); |
3166 | COIDEF_Status(sys->con.cntvect, &slv9_conopt_status); |
3167 | COIDEF_Message(sys->con.cntvect, &asc_conopt_message); |
3168 | COIDEF_ErrMsg(sys->con.cntvect, &slv9_conopt_errmsg); |
3169 | COIDEF_Progress(sys->con.cntvect, &asc_conopt_progress); |
3170 | |
3171 | /** @TODO implement the following options as well... */ |
3172 | #if 0 |
3173 | ipsz[F2C(10)] = 0; /* output to file */ |
3174 | ipsz[F2C(11)] = 1; /* progress info to screen */ |
3175 | ipsz[F2C(12)] = 1; /* correct value of func in coirms */ |
3176 | ipsz[F2C(13)] = 0; /* not correct value of jacs in coirms */ |
3177 | ipsz[F2C(14)] = 0; /* status not known by modeler */ |
3178 | ipsz[F2C(15)] = 0; /* function value include only NL terms */ |
3179 | ipsz[F2C(16)] = 1; /* Objective is a constraint */ |
3180 | ipsz[F2C(17)] = 0; /* sorted order for jacobian */ |
3181 | ipsz[F2C(18)] = 0; /* append the log file after restarts */ |
3182 | ipsz[F2C(19)] = 0; /* one subroutine call to coirms */ |
3183 | ipsz[F2C(20)] = 0; /* eval subroutine is coifde */ |
3184 | ipsz[F2C(21)] = 0; /* no debugging of derivatives */ |
3185 | ipsz[F2C(22)] = 0; /* coifde not called for linear eqns */ |
3186 | /* |
3187 | * skipping remainder of ipsz which are fortran io parameters |
3188 | */ |
3189 | |
3190 | /* |
3191 | * Real array |
3192 | */ |
3193 | rpsz[F2C(1)] = ASC_INFINITY; /* infinity */ |
3194 | rpsz[F2C(2)] = -ASC_INFINITY; /* -infinity */ |
3195 | rpsz[F2C(3)] = UNDEFINED; /* undefined */ |
3196 | rpsz[F2C(6)] = 0; /* work space allocated by conopt */ |
3197 | rpsz[F2C(7)] = TIME_LIMIT; /* resource limit (time) */ |
3198 | rpsz[F2C(8)] = 1; /* initial value for vars if none given */ |
3199 | #endif |
3200 | |
3201 | /* |
3202 | * reset count on coiopt calls |
3203 | */ |
3204 | sys->con.opt_count = 0; |
3205 | |
3206 | /* |
3207 | * do not keep model in memory after solution |
3208 | */ |
3209 | sys->con.kept = 0; |
3210 | |
3211 | COI_Solve(sys->con.cntvect); |
3212 | /* conopt_start(&(sys->con.kept), usrmem, &(sys->con.lwork), |
3213 | sys->con.work, &(sys->con.maxusd), &(sys->con.curusd)); */ |
3214 | |
3215 | /* |
3216 | * We assume that we get convergence in optimization problem at |
3217 | * boundary |
3218 | */ |
3219 | sys->con.optimized = 1; |
3220 | } |
3221 | |
3222 | #endif /* ASC_WITH_CONOPT */ |
3223 | |
3224 | /*-------------------end of conopt callbacks----------------------------------*/ |
3225 | |
3226 | |
3227 | /* |
3228 | * Creates an array of columns (containing an array of real elements |
3229 | * each) to storage the linear coefficient matrix of the optimization problem. |
3230 | * It also creates the arrays of reals required to storage the values |
3231 | * of the gradients of a subregion, which change depending on whether the |
3232 | * problem is a simulation or an optimization. |
3233 | */ |
3234 | static |
3235 | void create_opt_matrix_and_vectors(int32 num_opt_eqns, |
3236 | int32 n_subregions, |
3237 | struct opt_matrix *coeff_matrix, |
3238 | struct opt_vector *opt_var_values, |
3239 | struct opt_vector *invariant, |
3240 | struct opt_vector *variant, |
3241 | struct opt_vector *gradient, |
3242 | struct opt_matrix *multipliers |
3243 | ){ |
3244 | int32 c; |
3245 | int32 num_vars; |
3246 | |
3247 | num_vars = num_opt_eqns - 1 + n_subregions; |
3248 | |
3249 | coeff_matrix->cols = ASC_NEW_ARRAY(struct opt_vector,n_subregions); |
3250 | |
3251 | if(g_optimizing) { |
3252 | multipliers->cols = ASC_NEW_ARRAY(struct opt_vector,n_subregions); |
3253 | } |
3254 | |
3255 | for (c=0; c<n_subregions; c++) { |
3256 | coeff_matrix->cols[c].element = ASC_NEW_ARRAY(real64,num_opt_eqns); |
3257 | } |
3258 | opt_var_values->element = ASC_NEW_ARRAY(real64,num_vars); |
3259 | |
3260 | if(g_optimizing) { |
3261 | gradient->element = ASC_NEW_ARRAY(real64,num_opt_eqns); |
3262 | }else{ |
3263 | invariant->element = ASC_NEW_ARRAY(real64,num_opt_eqns); |
3264 | variant->element = ASC_NEW_ARRAY(real64,num_opt_eqns); |
3265 | } |
3266 | } |
3267 | |
3268 | |
3269 | /* |
3270 | * destroy the arrays created to storage the gradients for the optimization |
3271 | * problem |
3272 | */ |
3273 | static |
3274 | void destroy_opt_matrix_and_vectors(int32 n_subregions, |
3275 | struct opt_matrix *coeff_matrix, |
3276 | struct opt_vector *opt_var_values, |
3277 | struct opt_vector *invariant, |
3278 | struct opt_vector *variant, |
3279 | struct opt_vector *gradient, |
3280 | struct opt_matrix *multipliers |
3281 | ){ |
3282 | int32 c; |
3283 | for (c=0; c<n_subregions; c++) { |
3284 | destroy_array(coeff_matrix->cols[c].element); |
3285 | if(g_optimizing) { |
3286 | destroy_array(multipliers->cols[c].element); |
3287 | } |
3288 | } |
3289 | destroy_array(coeff_matrix->cols); |
3290 | destroy_array(opt_var_values->element); |
3291 | if(g_optimizing) { |
3292 | destroy_array(multipliers->cols); |
3293 | destroy_array(gradient->element); |
3294 | }else{ |
3295 | destroy_array(invariant->element); |
3296 | destroy_array(variant->element); |
3297 | } |
3298 | } |
3299 | |
3300 | |
3301 | |
3302 | /* |
3303 | * Set Factorization Options |
3304 | */ |
3305 | static |
3306 | void set_factor_options (linsolqr_system_t lsys){ |
3307 | linsolqr_prep(lsys,linsolqr_fmethod_to_fclass(ranki_ba2)); |
3308 | linsolqr_set_pivot_zero(lsys, 1e-12); |
3309 | linsolqr_set_drop_tolerance(lsys,1e-16); |
3310 | linsolqr_set_pivot_tolerance(lsys, 0.1); |
3311 | linsolqr_set_condition_tolerance(lsys, 0.1); |
3312 | } |
3313 | |
3314 | |
3315 | /* |
3316 | * Calculating the Lagrange Multipliers for each subregion |
3317 | * |
3318 | * We are assuming here that the matrix is structurally nonsingular |
3319 | * and than the rank is equal to the number of rows in the matrix. |
3320 | * Much more efficient checking must be done. |
3321 | */ |
3322 | static |
3323 | void get_multipliers(SlvClientToken asys, |
3324 | int32 subregion, |
3325 | int32 nrel, |
3326 | real64 *grad_obj, |
3327 | struct opt_matrix *multipliers |
3328 | ){ |
3329 | slv9_system_t sys; |
3330 | linsolqr_system_t lsys; |
3331 | mtx_region_t *newblocks, *oneblock; |
3332 | int32 rank; |
3333 | int32 c, cr, len, row; |
3334 | real64 *weights; |
3335 | real64 summ; |
3336 | |
3337 | sys = SLV9(asys); |
3338 | check_system(sys); |
3339 | |
3340 | mtx_output_assign(sys->lin_mtx,nrel,nrel); |
3341 | rank = mtx_symbolic_rank(sys->lin_mtx); |
3342 | mtx_partition(sys->lin_mtx); |
3343 | len = mtx_number_of_blocks(sys->lin_mtx); |
3344 | newblocks = ASC_NEW_ARRAY(mtx_region_t,len); |
3345 | if(newblocks == NULL) { |
3346 | mtx_destroy(sys->lin_mtx); |
3347 | return; |
3348 | } |
3349 | for (c = 0 ; c < len; c++) { |
3350 | mtx_block(sys->lin_mtx,c,&(newblocks[c])); |
3351 | } |
3352 | for (c = 0 ; c < len; c++) { |
3353 | mtx_reorder(sys->lin_mtx,&(newblocks[c]),mtx_SPK1); |
3354 | } |
3355 | |
3356 | /* unifying block */ |
3357 | oneblock = (mtx_region_t *)ascmalloc(sizeof(mtx_region_t)); |
3358 | oneblock->row.low = oneblock->col.low = 0; |
3359 | oneblock->row.high = nrel-1; |
3360 | oneblock->col.high = nrel-1; |
3361 | |
3362 | /* |
3363 | * Scaling of the linear system |
3364 | */ |
3365 | |
3366 | /* |
3367 | *Calculating weights |
3368 | */ |
3369 | weights = ASC_NEW_ARRAY(real64,nrel); |
3370 | for (row=0; row<nrel; row++) { |
3371 | summ = mtx_sum_sqrs_in_row(sys->lin_mtx,row,&(oneblock->col)); |
3372 | if(summ <= 0.0) { |
3373 | weights[row] = 1.0; |
3374 | }else{ |
3375 | weights[row] = 1.0 / sqrt(summ); |
3376 | } |
3377 | #if DEBUG |
3378 | FPRINTF(ASCERR," weight of row %d = %f \n",row,summ); |
3379 | #endif /* DEBUG */ |
3380 | } |
3381 | |
3382 | /* |
3383 | * Dividing rows by weights |
3384 | */ |
3385 | for (row=0; row<nrel; row++) { |
3386 | mtx_mult_row(sys->lin_mtx,row,weights[row],&(oneblock->col)); |
3387 | } |
3388 | |
3389 | /* |
3390 | * dividing rhs |
3391 | */ |
3392 | for (row=0; row<nrel; row++) { |
3393 | grad_obj[mtx_row_to_org(sys->lin_mtx,row)] = |
3394 | grad_obj[mtx_row_to_org(sys->lin_mtx,row)] * weights[row]; |
3395 | } |
3396 | |
3397 | /* |
3398 | * End of scaling |
3399 | */ |
3400 | |
3401 | lsys = linsolqr_create(); |
3402 | linsolqr_set_matrix(lsys,sys->lin_mtx); |
3403 | |
3404 | for (cr=0; cr<nrel; cr++) { |
3405 | multipliers->cols[subregion].element[cr] = 0.0; |
3406 | } |
3407 | |
3408 | set_factor_options(lsys); |
3409 | /* rhs for multipliers */ |
3410 | linsolqr_add_rhs(lsys,grad_obj,FALSE); |
3411 | linsolqr_set_region(lsys,*oneblock); |
3412 | linsolqr_factor(lsys,ranki_ba2); |
3413 | linsolqr_solve(lsys,grad_obj); |
3414 | for (cr=0; cr<nrel; cr++) { |
3415 | multipliers->cols[subregion].element[cr] = linsolqr_var_value |
3416 | (lsys,grad_obj,cr); |
3417 | #if SHOW_LAGRANGE_DETAILS |
3418 | FPRINTF(ASCERR, " Row = %d \n",cr); |
3419 | FPRINTF(ASCERR, |
3420 | "Multiplier = %f \n",multipliers->cols[subregion].element[cr]); |
3421 | #endif /* SHOW_LAGRANGE_DETAILS */ |
3422 | } |
3423 | linsolqr_set_matrix(lsys,NULL); |
3424 | mtx_destroy(sys->lin_mtx); |
3425 | linsolqr_destroy(lsys); |
3426 | destroy_array(newblocks); |
3427 | destroy_array(weights); |
3428 | ascfree(oneblock); |
3429 | } |
3430 | |
3431 | |
3432 | /* |
3433 | * Calculate the invariant part of the gradients of the subregions |
3434 | */ |
3435 | static |
3436 | void get_gradient_in_subregion(slv_system_t server, |
3437 | SlvClientToken asys, |
3438 | int32 subregion, |
3439 | int32 num_opt_eqns, |
3440 | struct opt_vector *gradient, |
3441 | struct opt_matrix *multipliers |
3442 | ){ |
3443 | slv9_system_t sys; |
3444 | struct rel_relation **rlist; |
3445 | struct var_variable **vlist; |
3446 | struct rel_relation *rel; |
3447 | struct var_variable *var; |
3448 | var_filter_t vfilter; |
3449 | rel_filter_t rfilter; |
3450 | mtx_coord_t coord; |
3451 | real64 *tmp_value; |
3452 | real64 *derivatives, resid; |
3453 | real64 *grad_obj, *func_val; |
3454 | real64 *f_red_grad; |
3455 | struct opt_matrix rel_red_grad; |
3456 | int32 *variables_master, *variables_solver, count; |
3457 | int32 nvar, nrel, ntotvar, ntotrel; |
3458 | int32 countrel,countvar,cr,cv,len,vind; |
3459 | int32 nvnb, countnbv; |
3460 | FILE *lif; |
3461 | |
3462 | sys = SLV9(asys); |
3463 | check_system(sys); |
3464 | lif = LIF(sys); |
3465 | |
3466 | rlist = slv_get_master_rel_list(server); |
3467 | vlist = slv_get_master_var_list(server); |
3468 | ntotvar = slv_get_num_master_vars(server); |
3469 | ntotrel = slv_get_num_master_rels(server); |
3470 | tmp_value = ASC_NEW_ARRAY(real64,ntotvar); |
3471 | |
3472 | vfilter.matchbits = (VAR_ACTIVE | VAR_INCIDENT | VAR_NONBASIC |
3473 | | VAR_SVAR | VAR_FIXED); |
3474 | vfilter.matchvalue = (VAR_ACTIVE | VAR_INCIDENT | VAR_SVAR); |
3475 | nvar = slv_count_master_vars(server,&vfilter); |
3476 | |
3477 | rfilter.matchbits = (REL_INCLUDED | REL_EQUALITY | REL_ACTIVE ); |
3478 | rfilter.matchvalue =(REL_INCLUDED | REL_EQUALITY | REL_ACTIVE ); |
3479 | nrel = slv_count_master_rels(server,&rfilter); |
3480 | |
3481 | if(nrel != nvar) { |
3482 | FPRINTF(ASCERR," nrel = %d\n",nrel); |
3483 | FPRINTF(ASCERR," nvar = %d\n",nvar); |
3484 | FPRINTF(ASCERR, |
3485 | "PANIC: number relations does not match number of variables\n"); |
3486 | } |
3487 | |
3488 | /* |
3489 | * residual of the relations in the subregion |
3490 | */ |
3491 | func_val = ASC_NEW_ARRAY(real64,nrel); |
3492 | |
3493 | /* |
3494 | * Lagrange Multipliers of the subregion |
3495 | */ |
3496 | multipliers->cols[subregion].element = |
3497 | (real64 *)(ascmalloc(nrel*sizeof(real64))); |
3498 | /* |
3499 | * Gradients of the objective function |
3500 | */ |
3501 | grad_obj = ASC_NEW_ARRAY(real64,nvar); |
3502 | |
3503 | /* |
3504 | * Matrix for solving linear system |
3505 | */ |
3506 | sys->lin_mtx = mtx_create(); |
3507 | mtx_set_order(sys->lin_mtx,nrel); |
3508 | |
3509 | /* |
3510 | * Counting nonbasic variables |
3511 | */ |
3512 | vfilter.matchbits = (VAR_ACTIVE | VAR_INCIDENT | VAR_NONBASIC |
3513 | | VAR_SVAR | VAR_FIXED); |
3514 | vfilter.matchvalue = (VAR_ACTIVE | VAR_INCIDENT | VAR_NONBASIC |
3515 | | VAR_SVAR); |
3516 | nvnb = slv_count_master_vars(server,&vfilter); |
3517 | |
3518 | /* |
3519 | * Information for reduced gradient |
3520 | */ |
3521 | f_red_grad = ASC_NEW_ARRAY(real64,nvnb); |
3522 | rel_red_grad.cols = (struct opt_vector *) |
3523 | (ascmalloc(nvnb*sizeof(struct opt_vector))); |
3524 | for (cv=0; cv<nvnb; cv++) { |
3525 | rel_red_grad.cols[cv].element = |
3526 | (real64 *)(ascmalloc(nrel*sizeof(real64))); |
3527 | } |
3528 | |
3529 | /* |
3530 | * Setting every sindex to -1 |
3531 | */ |
3532 | for (cv=0; cv<ntotvar; cv++) { |
3533 | var_set_sindex(vlist[cv],-1); |
3534 | } |
3535 | for (cr=0; cr<ntotrel; cr++) { |
3536 | rel_set_sindex(rlist[cr],-1); |
3537 | } |
3538 | |
3539 | /* |
3540 | * Initializing values |
3541 | */ |
3542 | for (cv=0; cv<ntotvar;cv++) { |
3543 | tmp_value[cv] = 0.0; |
3544 | } |
3545 | for (cv=0; cv<num_opt_eqns;cv++) { |
3546 | gradient->element[cv] = 0.0; |
3547 | } |
3548 | |
3549 | for (cr=0; cr<nrel; cr++) { |
3550 | func_val[cr] = 0.0; |
3551 | multipliers->cols[subregion].element[cr] = 0.0; |
3552 | } |
3553 | |
3554 | for (cv=0; cv<nvnb; cv++) { |
3555 | f_red_grad[cv] = 0.0; |
3556 | for (cr=0; cr<nrel; cr++) { |
3557 | rel_red_grad.cols[cv].element[cr] = 0.0; |
3558 | } |
3559 | } |
3560 | |
3561 | for (cv=0; cv<nvar; cv++) { |
3562 | grad_obj[cv] = 0.0; |
3563 | } |
3564 | |
3565 | /* |
3566 | * Calculate Values |
3567 | */ |
3568 | vfilter.matchbits = (VAR_ACTIVE_AT_BND | VAR_INCIDENT |
3569 | | VAR_SVAR | VAR_FIXED); |
3570 | vfilter.matchvalue = (VAR_ACTIVE_AT_BND | VAR_INCIDENT | VAR_SVAR); |
3571 | |
3572 | /* |
3573 | * List of relations |
3574 | */ |
3575 | countrel = 0; |
3576 | countvar = 0; |
3577 | countnbv = 0; |
3578 | for (cr=0; cr<ntotrel; cr++) { |
3579 | rel = rlist[cr]; |
3580 | if(rel_apply_filter(rel,&rfilter)) { |
3581 | rel_set_sindex(rel,countrel); |
3582 | coord.col = rel_sindex(rel); |
3583 | len = rel_n_incidences(rel); |
3584 | variables_master = ASC_NEW_ARRAY(int32,len); |
3585 | variables_solver = ASC_NEW_ARRAY(int32,len); |
3586 | derivatives = ASC_NEW_ARRAY(real64,len); |
3587 | relman_diff_grad(rel,&vfilter,derivatives,variables_master, |
3588 | variables_solver,&count,&resid,1); |
3589 | func_val[countrel] = resid; |
3590 | #if SHOW_LAGRANGE_DETAILS |
3591 | FPRINTF(ASCERR,"Equation = %d \n",coord.col); |
3592 | FPRINTF(ASCERR,"Residual = %f \n",resid); |
3593 | #endif /* SHOW_LAGRANGE_DETAILS */ |
3594 | for (cv=0; cv<count;cv++) { |
3595 | var = vlist[variables_master[cv]]; |
3596 | if(!var_nonbasic(var)) { |
3597 | tmp_value[variables_master[cv]] = tmp_value[variables_master[cv]] + |
3598 | derivatives[cv] * RHO * resid; |
3599 | if(var_active(var)) { |
3600 | coord.row = var_sindex(var); |
3601 | if(coord.row == -1) { |
3602 | var_set_sindex(var,countvar); |
3603 | coord.row = countvar; |
3604 | countvar++; |
3605 | } |
3606 | assert(coord.col >= 0 && coord.col < mtx_order(sys->lin_mtx)); |
3607 | #if SHOW_LAGRANGE_DETAILS |
3608 | FPRINTF(ASCERR,"Coordinate row = %d \n",coord.row); |
3609 | FPRINTF(ASCERR,"Coordinate col = %d \n",coord.col); |
3610 | FPRINTF(ASCERR,"Derivative = %f \n",derivatives[cv]); |
3611 | #endif /* SHOW_LAGRANGE_DETAILS */ |
3612 | mtx_fill_org_value(sys->lin_mtx,&coord,derivatives[cv]); |
3613 | } |
3614 | }else{ |
3615 | if(var_sindex(var)== -1) { |
3616 | var_set_sindex(var,countnbv); |
3617 | countnbv++; |
3618 | } |
3619 | rel_red_grad.cols[var_sindex(var)].element[countrel] = |
3620 | derivatives[cv]; |
3621 | #if SHOW_LAGRANGE_DETAILS |
3622 | FPRINTF(lif,"Nonbasic Variable "); |
3623 | print_var_name(lif,sys,var); PUTC('\n',lif); |
3624 | FPRINTF(ASCERR,"Derivative = %f \n",derivatives[cv]); |
3625 | #endif /* SHOW_LAGRANGE_DETAILS */ |
3626 | } |
3627 | } |
3628 | destroy_array(variables_master); |
3629 | destroy_array(variables_solver); |
3630 | destroy_array(derivatives); |
3631 | countrel++; |
3632 | } |
3633 | } |
3634 | |
3635 | /* |
3636 | * Objective function |
3637 | */ |
3638 | rel = sys->obj; |
3639 | len = rel_n_incidences(rel); |
3640 | variables_master = ASC_NEW_ARRAY |